xref: /openbmc/openpower-vpd-parser/vpd-manager/src/single_fab.cpp (revision 57022297e494963ffc04d23d20e9e415971c6577)
1b5bfcbc6SAnupama B R #include "config.h"
2b5bfcbc6SAnupama B R 
308fa59eaSAnupama B R #include "single_fab.hpp"
408fa59eaSAnupama B R 
508fa59eaSAnupama B R #include "constants.hpp"
6*57022297SRekha Aparna #include "logger.hpp"
7d3e693ebSRekhaAparna01 #include "parser.hpp"
808fa59eaSAnupama B R #include "types.hpp"
908fa59eaSAnupama B R 
10d3e693ebSRekhaAparna01 #include <nlohmann/json.hpp>
11f05f354dSRekhaAparna01 #include <utility/common_utility.hpp>
12a39aafa3SRekha Aparna #include <utility/event_logger_utility.hpp>
1308fa59eaSAnupama B R #include <utility/json_utility.hpp>
14ffdff313SRekha Aparna #include <utility/vpd_specific_utility.hpp>
1508fa59eaSAnupama B R 
1608fa59eaSAnupama B R namespace vpd
1708fa59eaSAnupama B R {
1808fa59eaSAnupama B R constexpr auto pimPersistVsbpPath =
1908fa59eaSAnupama B R     "/var/lib/phosphor-inventory-manager/xyz/openbmc_project/inventory/system/chassis/motherboard/com.ibm.ipzvpd.VSBP";
20b5bfcbc6SAnupama B R constexpr auto IM_SIZE_IN_BYTES = 0x04;
21b5bfcbc6SAnupama B R constexpr auto IM_KW_VALUE_OFFSET = 0x000005fb;
2208fa59eaSAnupama B R 
getImFromPersistedLocation() const2308fa59eaSAnupama B R std::string SingleFab::getImFromPersistedLocation() const noexcept
2408fa59eaSAnupama B R {
2508fa59eaSAnupama B R     try
2608fa59eaSAnupama B R     {
27ca9a0866SRekha Aparna         uint16_t l_errCode = 0;
2808fa59eaSAnupama B R         auto l_parsedVsbpJsonObj =
29ca9a0866SRekha Aparna             jsonUtility::getParsedJson(pimPersistVsbpPath, l_errCode);
30ca9a0866SRekha Aparna 
31ca9a0866SRekha Aparna         if (l_errCode)
32ca9a0866SRekha Aparna         {
33ca9a0866SRekha Aparna             throw JsonException(
34ca9a0866SRekha Aparna                 "Failed to parse JSON file [ " +
35c6159a29SRekha Aparna                     std::string(pimPersistVsbpPath) +
36c6159a29SRekha Aparna                     " ], error : " + commonUtility::getErrCodeMsg(l_errCode),
37ca9a0866SRekha Aparna                 pimPersistVsbpPath);
38ca9a0866SRekha Aparna         }
39ca9a0866SRekha Aparna 
4008fa59eaSAnupama B R         if (!l_parsedVsbpJsonObj.contains("value0") ||
4108fa59eaSAnupama B R             !l_parsedVsbpJsonObj["value0"].contains(constants::kwdIM) ||
4208fa59eaSAnupama B R             !l_parsedVsbpJsonObj["value0"][constants::kwdIM].is_array())
4308fa59eaSAnupama B R         {
44ca9a0866SRekha Aparna             throw std::runtime_error("Mandatory tag(s) missing from JSON");
4508fa59eaSAnupama B R         }
4608fa59eaSAnupama B R 
4708fa59eaSAnupama B R         const types::BinaryVector l_imValue =
4808fa59eaSAnupama B R             l_parsedVsbpJsonObj["value0"][constants::kwdIM]
4908fa59eaSAnupama B R                 .get<types::BinaryVector>();
5008fa59eaSAnupama B R 
5108fa59eaSAnupama B R         std::ostringstream l_imData;
5208fa59eaSAnupama B R         for (const auto& l_byte : l_imValue)
5308fa59eaSAnupama B R         {
5408fa59eaSAnupama B R             l_imData << std::setw(2) << std::setfill('0') << std::hex
5508fa59eaSAnupama B R                      << static_cast<int>(l_byte);
5608fa59eaSAnupama B R         }
5708fa59eaSAnupama B R         return l_imData.str();
5808fa59eaSAnupama B R     }
5908fa59eaSAnupama B R     catch (const std::exception& l_ex)
60d3e693ebSRekhaAparna01     {}
6108fa59eaSAnupama B R 
6208fa59eaSAnupama B R     return std::string();
6308fa59eaSAnupama B R }
64b5bfcbc6SAnupama B R 
getImFromPlanar() const65b5bfcbc6SAnupama B R std::string SingleFab::getImFromPlanar() const noexcept
66b5bfcbc6SAnupama B R {
67b5bfcbc6SAnupama B R     try
68b5bfcbc6SAnupama B R     {
69e2f09be8SAnupama B R         const std::string l_systemPlanarPath(SYSTEM_VPD_FILE_PATH);
70e2f09be8SAnupama B R         Parser l_parserObj(l_systemPlanarPath, nlohmann::json{});
71b5bfcbc6SAnupama B R 
72e2f09be8SAnupama B R         std::shared_ptr<ParserInterface> l_vpdParserInstance =
73e2f09be8SAnupama B R             l_parserObj.getVpdParserInstance();
74b5bfcbc6SAnupama B R 
75e2f09be8SAnupama B R         auto l_readValue = l_vpdParserInstance->readKeywordFromHardware(
76e2f09be8SAnupama B R             std::make_tuple(constants::recVSBP, constants::kwdIM));
77b5bfcbc6SAnupama B R 
78e2f09be8SAnupama B R         if (auto l_keywordValue =
79e2f09be8SAnupama B R                 std::get_if<types::BinaryVector>(&l_readValue);
80e2f09be8SAnupama B R             l_keywordValue && !l_keywordValue->empty())
81e2f09be8SAnupama B R         {
82b5bfcbc6SAnupama B R             std::ostringstream l_imData;
83e2f09be8SAnupama B R             for (const auto& l_byte : *l_keywordValue)
84b5bfcbc6SAnupama B R             {
85b5bfcbc6SAnupama B R                 l_imData << std::setw(2) << std::setfill('0') << std::hex
86b5bfcbc6SAnupama B R                          << static_cast<int>(l_byte);
87b5bfcbc6SAnupama B R             }
88e2f09be8SAnupama B R 
89b5bfcbc6SAnupama B R             return l_imData.str();
90b5bfcbc6SAnupama B R         }
91e2f09be8SAnupama B R     }
92b5bfcbc6SAnupama B R     catch (const std::ifstream::failure& l_ex)
93b5bfcbc6SAnupama B R     {}
94b5bfcbc6SAnupama B R 
95b5bfcbc6SAnupama B R     return std::string();
96b5bfcbc6SAnupama B R }
97d3e693ebSRekhaAparna01 
setImOnPlanar(const std::string & i_imValue) const98d3e693ebSRekhaAparna01 bool SingleFab::setImOnPlanar(const std::string& i_imValue) const noexcept
99d3e693ebSRekhaAparna01 {
100d3e693ebSRekhaAparna01     try
101d3e693ebSRekhaAparna01     {
102d3e693ebSRekhaAparna01         types::BinaryVector l_imValue;
103d3e693ebSRekhaAparna01         const std::string l_systemPlanarEepromPath = SYSTEM_VPD_FILE_PATH;
104d3e693ebSRekhaAparna01 
105d3e693ebSRekhaAparna01         // Convert string to vector of bytes
106d3e693ebSRekhaAparna01         for (auto l_value : i_imValue | std::views::chunk(2))
107d3e693ebSRekhaAparna01         {
108d3e693ebSRekhaAparna01             std::string l_byteString(l_value.begin(), l_value.end());
109d3e693ebSRekhaAparna01             l_imValue.push_back(
110d3e693ebSRekhaAparna01                 static_cast<uint8_t>(std::stoi(l_byteString, nullptr, 16)));
111d3e693ebSRekhaAparna01         }
112d3e693ebSRekhaAparna01 
113d3e693ebSRekhaAparna01         std::shared_ptr<Parser> l_parserObj = std::make_shared<Parser>(
114d3e693ebSRekhaAparna01             l_systemPlanarEepromPath, nlohmann::json{});
115d3e693ebSRekhaAparna01 
116d3e693ebSRekhaAparna01         int l_bytes_updated = l_parserObj->updateVpdKeywordOnHardware(
117d3e693ebSRekhaAparna01             std::make_tuple(constants::recVSBP, constants::kwdIM, l_imValue));
118d3e693ebSRekhaAparna01 
119d3e693ebSRekhaAparna01         return l_bytes_updated > 0 ? true : false;
120d3e693ebSRekhaAparna01     }
121d3e693ebSRekhaAparna01     catch (const std::exception& l_ex)
122d3e693ebSRekhaAparna01     {
123d3e693ebSRekhaAparna01         return false;
124d3e693ebSRekhaAparna01     }
125d3e693ebSRekhaAparna01 }
126f05f354dSRekhaAparna01 
updateSystemImValueInVpdToP11Series(std::string i_currentImValuePlanar) const127ffdff313SRekha Aparna void SingleFab::updateSystemImValueInVpdToP11Series(
128cd828d48SSouvik Roy     std::string i_currentImValuePlanar) const noexcept
129cd828d48SSouvik Roy {
130cd828d48SSouvik Roy     bool l_retVal{false};
131cd828d48SSouvik Roy     if (!i_currentImValuePlanar.empty())
132cd828d48SSouvik Roy     {
133879d465bSAnupama B R         if (i_currentImValuePlanar.compare(
134879d465bSAnupama B R                 constants::VALUE_4, constants::VALUE_1,
135879d465bSAnupama B R                 std::to_string(constants::VALUE_3)) ==
136879d465bSAnupama B R             constants::STR_CMP_SUCCESS)
137879d465bSAnupama B R         {
138879d465bSAnupama B R             i_currentImValuePlanar.replace(constants::VALUE_4,
139879d465bSAnupama B R                                            constants::VALUE_1,
140879d465bSAnupama B R                                            std::to_string(constants::VALUE_2));
141879d465bSAnupama B R         }
142879d465bSAnupama B R 
143cd828d48SSouvik Roy         // update the IM value to P11 series(6000x). Replace the first character
144cd828d48SSouvik Roy         // of IM value string with '6'
145cd828d48SSouvik Roy         l_retVal = setImOnPlanar(i_currentImValuePlanar.replace(
146cd828d48SSouvik Roy             constants::VALUE_0, constants::VALUE_1,
147cd828d48SSouvik Roy             std::to_string(constants::VALUE_6)));
148cd828d48SSouvik Roy     }
149ffdff313SRekha Aparna 
150ffdff313SRekha Aparna     if (!l_retVal)
151ffdff313SRekha Aparna     {
152ffdff313SRekha Aparna         EventLogger::createSyncPel(
153ffdff313SRekha Aparna             types::ErrorType::InternalFailure,
154ffdff313SRekha Aparna             types::SeverityType::Informational, __FILE__, __FUNCTION__, 0,
155ffdff313SRekha Aparna             std::string("Failed to update IM value to P11 series."),
156ffdff313SRekha Aparna             std::nullopt, std::nullopt, std::nullopt, std::nullopt);
157ffdff313SRekha Aparna     }
158ffdff313SRekha Aparna }
159ffdff313SRekha Aparna 
singleFabImOverride() const160ffdff313SRekha Aparna int SingleFab::singleFabImOverride() const noexcept
161ffdff313SRekha Aparna {
162*57022297SRekha Aparna     uint16_t l_errCode = 0;
163ffdff313SRekha Aparna     const std::string& l_planarImValue = getImFromPlanar();
164ffdff313SRekha Aparna     const std::string& l_eBmcImValue = getImFromPersistedLocation();
165*57022297SRekha Aparna     const bool& l_isFieldModeEnabled =
166*57022297SRekha Aparna         commonUtility::isFieldModeEnabled(l_errCode);
167*57022297SRekha Aparna 
168*57022297SRekha Aparna     if (l_errCode)
169*57022297SRekha Aparna     {
170*57022297SRekha Aparna         Logger::getLoggerInstance()->logMessage(
171*57022297SRekha Aparna             "Failed to check is field mode enabled, error : " +
172*57022297SRekha Aparna             commonUtility::getErrCodeMsg(l_errCode));
173*57022297SRekha Aparna     }
174*57022297SRekha Aparna 
175ffdff313SRekha Aparna     const bool& l_isLabModeEnabled =
176ffdff313SRekha Aparna         !l_isFieldModeEnabled;                       // used for understanding
177ffdff313SRekha Aparna     const bool& l_isPowerVsImage = vpdSpecificUtility::isPowerVsImage();
178ffdff313SRekha Aparna     const bool& l_isNormalImage = !l_isPowerVsImage; // used for understanding
179ffdff313SRekha Aparna 
180ffdff313SRekha Aparna     if (!isValidImSeries(l_planarImValue))
181ffdff313SRekha Aparna     {
182ffdff313SRekha Aparna         // Create Errorlog for invalid IM series encountered
183ffdff313SRekha Aparna         EventLogger::createSyncPel(
184ffdff313SRekha Aparna             types::ErrorType::InvalidSystem, types::SeverityType::Error,
185ffdff313SRekha Aparna             __FILE__, __FUNCTION__, 0,
186ffdff313SRekha Aparna             std::string("Invalid IM found on the system planar, IM value : ") +
187ffdff313SRekha Aparna                 l_planarImValue,
188ffdff313SRekha Aparna             std::nullopt, std::nullopt, std::nullopt, std::nullopt);
189ffdff313SRekha Aparna 
190ffdff313SRekha Aparna         return constants::SUCCESS;
191ffdff313SRekha Aparna     }
192ffdff313SRekha Aparna 
193ffdff313SRekha Aparna     if (!l_eBmcImValue.empty())
194ffdff313SRekha Aparna     {
195ffdff313SRekha Aparna         if (isP10System(l_eBmcImValue))
196ffdff313SRekha Aparna         {
197ffdff313SRekha Aparna             if (isP10System(l_planarImValue))
198ffdff313SRekha Aparna             {
199ffdff313SRekha Aparna                 if (l_isFieldModeEnabled && l_isNormalImage)
200ffdff313SRekha Aparna                 {
201ffdff313SRekha Aparna                     EventLogger::createSyncPel(
202ffdff313SRekha Aparna                         types::ErrorType::SystemTypeMismatch,
203ffdff313SRekha Aparna                         types::SeverityType::Warning, __FILE__, __FUNCTION__, 0,
204ffdff313SRekha Aparna                         std::string("Mismatch in IM value found eBMC IM [") +
205ffdff313SRekha Aparna                             l_eBmcImValue + std::string("] planar IM [") +
206ffdff313SRekha Aparna                             l_planarImValue +
207ffdff313SRekha Aparna                             std::string("] Field mode enabled [") +
208ffdff313SRekha Aparna                             ((l_isFieldModeEnabled) ? "true" : "false") +
209ffdff313SRekha Aparna                             std::string("]"),
210ffdff313SRekha Aparna                         std::nullopt, std::nullopt, std::nullopt, std::nullopt);
211ffdff313SRekha Aparna 
212ffdff313SRekha Aparna                     return constants::FAILURE;
213ffdff313SRekha Aparna                 }
214ffdff313SRekha Aparna             }
215ffdff313SRekha Aparna             else if (isP11System(l_planarImValue))
216ffdff313SRekha Aparna             {
217ffdff313SRekha Aparna                 if (!(l_isLabModeEnabled && l_isNormalImage))
218ffdff313SRekha Aparna                 {
219ffdff313SRekha Aparna                     EventLogger::createSyncPel(
220ffdff313SRekha Aparna                         types::ErrorType::SystemTypeMismatch,
221ffdff313SRekha Aparna                         types::SeverityType::Warning, __FILE__, __FUNCTION__, 0,
222ffdff313SRekha Aparna                         std::string("Mismatch in IM value found eBMC IM [") +
223ffdff313SRekha Aparna                             l_eBmcImValue + std::string("] planar IM [") +
224ffdff313SRekha Aparna                             l_planarImValue +
225ffdff313SRekha Aparna                             std::string("] Field mode enabled [") +
226ffdff313SRekha Aparna                             ((l_isFieldModeEnabled) ? "true" : "false") +
227ffdff313SRekha Aparna                             std::string("]"),
228ffdff313SRekha Aparna                         std::nullopt, std::nullopt, std::nullopt, std::nullopt);
229ffdff313SRekha Aparna 
230ffdff313SRekha Aparna                     return constants::FAILURE;
231ffdff313SRekha Aparna                 }
232ffdff313SRekha Aparna             }
233ffdff313SRekha Aparna         }
234ffdff313SRekha Aparna         else if (isP11System(l_eBmcImValue))
235ffdff313SRekha Aparna         {
236ffdff313SRekha Aparna             if (l_isPowerVsImage)
237ffdff313SRekha Aparna             {
238ffdff313SRekha Aparna                 EventLogger::createSyncPel(
239ffdff313SRekha Aparna                     types::ErrorType::SystemTypeMismatch,
240ffdff313SRekha Aparna                     types::SeverityType::Warning, __FILE__, __FUNCTION__, 0,
241ffdff313SRekha Aparna                     std::string("Mismatch in IM value found eBMC IM [") +
242ffdff313SRekha Aparna                         l_eBmcImValue + std::string("] planar IM [") +
243ffdff313SRekha Aparna                         l_planarImValue +
244ffdff313SRekha Aparna                         std::string("] Field mode enabled [") +
245ffdff313SRekha Aparna                         ((l_isFieldModeEnabled) ? "true" : "false") +
246ffdff313SRekha Aparna                         std::string("]"),
247ffdff313SRekha Aparna                     std::nullopt, std::nullopt, std::nullopt, std::nullopt);
248ffdff313SRekha Aparna 
249ffdff313SRekha Aparna                 return constants::FAILURE;
250ffdff313SRekha Aparna             }
251ffdff313SRekha Aparna             else
252ffdff313SRekha Aparna             {
253ffdff313SRekha Aparna                 if (isP10System(l_planarImValue))
254ffdff313SRekha Aparna                 {
255ffdff313SRekha Aparna                     updateSystemImValueInVpdToP11Series(l_planarImValue);
256ffdff313SRekha Aparna                 }
257ffdff313SRekha Aparna             }
258ffdff313SRekha Aparna         }
259ffdff313SRekha Aparna     }
260ffdff313SRekha Aparna     else
261ffdff313SRekha Aparna     {
262ffdff313SRekha Aparna         if (isP11System(l_planarImValue) && l_isPowerVsImage)
263ffdff313SRekha Aparna         {
264ffdff313SRekha Aparna             EventLogger::createSyncPel(
265ffdff313SRekha Aparna                 types::ErrorType::SystemTypeMismatch,
266ffdff313SRekha Aparna                 types::SeverityType::Warning, __FILE__, __FUNCTION__, 0,
267ffdff313SRekha Aparna                 std::string("Mismatch in IM value found eBMC IM [") +
268ffdff313SRekha Aparna                     l_eBmcImValue + std::string("] planar IM [") +
269ffdff313SRekha Aparna                     l_planarImValue + std::string("] Field mode enabled [") +
270ffdff313SRekha Aparna                     ((l_isFieldModeEnabled) ? "true" : "false") +
271ffdff313SRekha Aparna                     std::string("]"),
272ffdff313SRekha Aparna                 std::nullopt, std::nullopt, std::nullopt, std::nullopt);
273ffdff313SRekha Aparna 
274ffdff313SRekha Aparna             return constants::FAILURE;
275ffdff313SRekha Aparna         }
276ffdff313SRekha Aparna         else if (isP10System(l_planarImValue) && l_isNormalImage)
277ffdff313SRekha Aparna         {
278ffdff313SRekha Aparna             if (l_isLabModeEnabled)
279ffdff313SRekha Aparna             {
280ffdff313SRekha Aparna                 EventLogger::createSyncPel(
281ffdff313SRekha Aparna                     types::ErrorType::UnknownSystemSettings,
282ffdff313SRekha Aparna                     types::SeverityType::Warning, __FILE__, __FUNCTION__, 0,
283ffdff313SRekha Aparna                     std::string("Mismatch in IM value found eBMC IM [") +
284ffdff313SRekha Aparna                         l_eBmcImValue + std::string("] planar IM [") +
285ffdff313SRekha Aparna                         l_planarImValue +
286ffdff313SRekha Aparna                         std::string("] Field mode enabled [") +
287ffdff313SRekha Aparna                         ((l_isFieldModeEnabled) ? "true" : "false") +
288ffdff313SRekha Aparna                         std::string("]"),
289ffdff313SRekha Aparna                     std::nullopt, std::nullopt, std::nullopt, std::nullopt);
290ffdff313SRekha Aparna             }
291ffdff313SRekha Aparna             else
292ffdff313SRekha Aparna             {
293ffdff313SRekha Aparna                 updateSystemImValueInVpdToP11Series(l_planarImValue);
294ffdff313SRekha Aparna             }
295ffdff313SRekha Aparna         }
296ffdff313SRekha Aparna     }
297ffdff313SRekha Aparna     return constants::SUCCESS;
298cd828d48SSouvik Roy }
29908fa59eaSAnupama B R } // namespace vpd
300