xref: /openbmc/phosphor-bmc-code-mgmt/bmc/version.cpp (revision cab87e9cdeeb3e166d6d577511f6be4dc7721aca)
1*cab87e9cSJagpal Singh Gill #include "config.h"
2*cab87e9cSJagpal Singh Gill 
3*cab87e9cSJagpal Singh Gill #include "version.hpp"
4*cab87e9cSJagpal Singh Gill 
5*cab87e9cSJagpal Singh Gill #include "xyz/openbmc_project/Common/error.hpp"
6*cab87e9cSJagpal Singh Gill 
7*cab87e9cSJagpal Singh Gill #include <openssl/evp.h>
8*cab87e9cSJagpal Singh Gill 
9*cab87e9cSJagpal Singh Gill #include <phosphor-logging/elog-errors.hpp>
10*cab87e9cSJagpal Singh Gill #include <phosphor-logging/lg2.hpp>
11*cab87e9cSJagpal Singh Gill 
12*cab87e9cSJagpal Singh Gill #include <fstream>
13*cab87e9cSJagpal Singh Gill #include <iostream>
14*cab87e9cSJagpal Singh Gill #include <sstream>
15*cab87e9cSJagpal Singh Gill #include <stdexcept>
16*cab87e9cSJagpal Singh Gill #include <string>
17*cab87e9cSJagpal Singh Gill 
18*cab87e9cSJagpal Singh Gill namespace phosphor
19*cab87e9cSJagpal Singh Gill {
20*cab87e9cSJagpal Singh Gill namespace software
21*cab87e9cSJagpal Singh Gill {
22*cab87e9cSJagpal Singh Gill namespace manager
23*cab87e9cSJagpal Singh Gill {
24*cab87e9cSJagpal Singh Gill 
25*cab87e9cSJagpal Singh Gill PHOSPHOR_LOG2_USING;
26*cab87e9cSJagpal Singh Gill using namespace phosphor::logging;
27*cab87e9cSJagpal Singh Gill using Argument = xyz::openbmc_project::common::InvalidArgument;
28*cab87e9cSJagpal Singh Gill using namespace sdbusplus::error::xyz::openbmc_project::common;
29*cab87e9cSJagpal Singh Gill 
getValue(const std::string & manifestFilePath,std::string key)30*cab87e9cSJagpal Singh Gill std::string Version::getValue(const std::string& manifestFilePath,
31*cab87e9cSJagpal Singh Gill                               std::string key)
32*cab87e9cSJagpal Singh Gill {
33*cab87e9cSJagpal Singh Gill     std::vector<std::string> values = getRepeatedValues(manifestFilePath, key);
34*cab87e9cSJagpal Singh Gill     if (values.empty())
35*cab87e9cSJagpal Singh Gill     {
36*cab87e9cSJagpal Singh Gill         return std::string{};
37*cab87e9cSJagpal Singh Gill     }
38*cab87e9cSJagpal Singh Gill     if (values.size() > 1)
39*cab87e9cSJagpal Singh Gill     {
40*cab87e9cSJagpal Singh Gill         error("Multiple values found in MANIFEST file for key: {KEY}", "KEY",
41*cab87e9cSJagpal Singh Gill               key);
42*cab87e9cSJagpal Singh Gill     }
43*cab87e9cSJagpal Singh Gill     return values.at(0);
44*cab87e9cSJagpal Singh Gill }
45*cab87e9cSJagpal Singh Gill 
getRepeatedValues(const std::string & manifestFilePath,std::string key)46*cab87e9cSJagpal Singh Gill std::vector<std::string> Version::getRepeatedValues(
47*cab87e9cSJagpal Singh Gill     const std::string& manifestFilePath, std::string key)
48*cab87e9cSJagpal Singh Gill {
49*cab87e9cSJagpal Singh Gill     key = key + "=";
50*cab87e9cSJagpal Singh Gill     auto keySize = key.length();
51*cab87e9cSJagpal Singh Gill 
52*cab87e9cSJagpal Singh Gill     if (manifestFilePath.empty())
53*cab87e9cSJagpal Singh Gill     {
54*cab87e9cSJagpal Singh Gill         error("ManifestFilePath is empty.");
55*cab87e9cSJagpal Singh Gill         elog<InvalidArgument>(
56*cab87e9cSJagpal Singh Gill             Argument::ARGUMENT_NAME("manifestFilePath"),
57*cab87e9cSJagpal Singh Gill             Argument::ARGUMENT_VALUE(manifestFilePath.c_str()));
58*cab87e9cSJagpal Singh Gill     }
59*cab87e9cSJagpal Singh Gill 
60*cab87e9cSJagpal Singh Gill     std::vector<std::string> values{};
61*cab87e9cSJagpal Singh Gill     std::ifstream efile;
62*cab87e9cSJagpal Singh Gill     std::string line;
63*cab87e9cSJagpal Singh Gill     efile.exceptions(std::ifstream::failbit | std::ifstream::badbit);
64*cab87e9cSJagpal Singh Gill 
65*cab87e9cSJagpal Singh Gill     // Too many GCC bugs (53984, 66145) to do this the right way...
66*cab87e9cSJagpal Singh Gill     try
67*cab87e9cSJagpal Singh Gill     {
68*cab87e9cSJagpal Singh Gill         efile.open(manifestFilePath);
69*cab87e9cSJagpal Singh Gill         while (getline(efile, line))
70*cab87e9cSJagpal Singh Gill         {
71*cab87e9cSJagpal Singh Gill             if (!line.empty() && line.back() == '\r')
72*cab87e9cSJagpal Singh Gill             {
73*cab87e9cSJagpal Singh Gill                 // If the manifest has CRLF line terminators, e.g. is created on
74*cab87e9cSJagpal Singh Gill                 // Windows, the line will contain \r at the end, remove it.
75*cab87e9cSJagpal Singh Gill                 line.pop_back();
76*cab87e9cSJagpal Singh Gill             }
77*cab87e9cSJagpal Singh Gill             if (line.compare(0, keySize, key) == 0)
78*cab87e9cSJagpal Singh Gill             {
79*cab87e9cSJagpal Singh Gill                 values.push_back(line.substr(keySize));
80*cab87e9cSJagpal Singh Gill             }
81*cab87e9cSJagpal Singh Gill         }
82*cab87e9cSJagpal Singh Gill         efile.close();
83*cab87e9cSJagpal Singh Gill     }
84*cab87e9cSJagpal Singh Gill     catch (const std::exception& e)
85*cab87e9cSJagpal Singh Gill     {
86*cab87e9cSJagpal Singh Gill         if (!efile.eof())
87*cab87e9cSJagpal Singh Gill         {
88*cab87e9cSJagpal Singh Gill             error("Error occurred when reading MANIFEST file: {ERROR}", "KEY",
89*cab87e9cSJagpal Singh Gill                   key, "ERROR", e);
90*cab87e9cSJagpal Singh Gill         }
91*cab87e9cSJagpal Singh Gill     }
92*cab87e9cSJagpal Singh Gill 
93*cab87e9cSJagpal Singh Gill     if (values.empty())
94*cab87e9cSJagpal Singh Gill     {
95*cab87e9cSJagpal Singh Gill         info("No values found in MANIFEST file for key: {KEY}", "KEY", key);
96*cab87e9cSJagpal Singh Gill     }
97*cab87e9cSJagpal Singh Gill 
98*cab87e9cSJagpal Singh Gill     return values;
99*cab87e9cSJagpal Singh Gill }
100*cab87e9cSJagpal Singh Gill 
101*cab87e9cSJagpal Singh Gill using EVP_MD_CTX_Ptr =
102*cab87e9cSJagpal Singh Gill     std::unique_ptr<EVP_MD_CTX, decltype(&::EVP_MD_CTX_free)>;
103*cab87e9cSJagpal Singh Gill 
getId(const std::string & version)104*cab87e9cSJagpal Singh Gill std::string Version::getId(const std::string& version)
105*cab87e9cSJagpal Singh Gill {
106*cab87e9cSJagpal Singh Gill     if (version.empty())
107*cab87e9cSJagpal Singh Gill     {
108*cab87e9cSJagpal Singh Gill         error("Version is empty.");
109*cab87e9cSJagpal Singh Gill         elog<InvalidArgument>(Argument::ARGUMENT_NAME("Version"),
110*cab87e9cSJagpal Singh Gill                               Argument::ARGUMENT_VALUE(version.c_str()));
111*cab87e9cSJagpal Singh Gill     }
112*cab87e9cSJagpal Singh Gill 
113*cab87e9cSJagpal Singh Gill     std::array<unsigned char, EVP_MAX_MD_SIZE> digest{};
114*cab87e9cSJagpal Singh Gill     EVP_MD_CTX_Ptr ctx(EVP_MD_CTX_new(), &::EVP_MD_CTX_free);
115*cab87e9cSJagpal Singh Gill 
116*cab87e9cSJagpal Singh Gill     EVP_DigestInit(ctx.get(), EVP_sha512());
117*cab87e9cSJagpal Singh Gill     EVP_DigestUpdate(ctx.get(), version.c_str(), strlen(version.c_str()));
118*cab87e9cSJagpal Singh Gill     EVP_DigestFinal(ctx.get(), digest.data(), nullptr);
119*cab87e9cSJagpal Singh Gill 
120*cab87e9cSJagpal Singh Gill     // We are only using the first 8 characters.
121*cab87e9cSJagpal Singh Gill     char mdString[9];
122*cab87e9cSJagpal Singh Gill     snprintf(mdString, sizeof(mdString), "%02x%02x%02x%02x",
123*cab87e9cSJagpal Singh Gill              (unsigned int)digest[0], (unsigned int)digest[1],
124*cab87e9cSJagpal Singh Gill              (unsigned int)digest[2], (unsigned int)digest[3]);
125*cab87e9cSJagpal Singh Gill 
126*cab87e9cSJagpal Singh Gill     return mdString;
127*cab87e9cSJagpal Singh Gill }
128*cab87e9cSJagpal Singh Gill 
getBMCMachine(const std::string & releaseFilePath)129*cab87e9cSJagpal Singh Gill std::string Version::getBMCMachine(const std::string& releaseFilePath)
130*cab87e9cSJagpal Singh Gill {
131*cab87e9cSJagpal Singh Gill     std::string machineKey = "OPENBMC_TARGET_MACHINE=";
132*cab87e9cSJagpal Singh Gill     std::string machine{};
133*cab87e9cSJagpal Singh Gill     std::ifstream efile(releaseFilePath);
134*cab87e9cSJagpal Singh Gill     std::string line;
135*cab87e9cSJagpal Singh Gill 
136*cab87e9cSJagpal Singh Gill     while (getline(efile, line))
137*cab87e9cSJagpal Singh Gill     {
138*cab87e9cSJagpal Singh Gill         if (line.substr(0, machineKey.size()).find(machineKey) !=
139*cab87e9cSJagpal Singh Gill             std::string::npos)
140*cab87e9cSJagpal Singh Gill         {
141*cab87e9cSJagpal Singh Gill             std::size_t pos = line.find_first_of('"') + 1;
142*cab87e9cSJagpal Singh Gill             machine = line.substr(pos, line.find_last_of('"') - pos);
143*cab87e9cSJagpal Singh Gill             break;
144*cab87e9cSJagpal Singh Gill         }
145*cab87e9cSJagpal Singh Gill     }
146*cab87e9cSJagpal Singh Gill 
147*cab87e9cSJagpal Singh Gill     if (machine.empty())
148*cab87e9cSJagpal Singh Gill     {
149*cab87e9cSJagpal Singh Gill         error("Unable to find OPENBMC_TARGET_MACHINE");
150*cab87e9cSJagpal Singh Gill         elog<InternalFailure>();
151*cab87e9cSJagpal Singh Gill     }
152*cab87e9cSJagpal Singh Gill 
153*cab87e9cSJagpal Singh Gill     return machine;
154*cab87e9cSJagpal Singh Gill }
155*cab87e9cSJagpal Singh Gill 
getBMCExtendedVersion(const std::string & releaseFilePath)156*cab87e9cSJagpal Singh Gill std::string Version::getBMCExtendedVersion(const std::string& releaseFilePath)
157*cab87e9cSJagpal Singh Gill {
158*cab87e9cSJagpal Singh Gill     std::string extendedVersionKey = "EXTENDED_VERSION=";
159*cab87e9cSJagpal Singh Gill     std::string extendedVersionValue{};
160*cab87e9cSJagpal Singh Gill     std::string extendedVersion{};
161*cab87e9cSJagpal Singh Gill     std::ifstream efile(releaseFilePath);
162*cab87e9cSJagpal Singh Gill     std::string line;
163*cab87e9cSJagpal Singh Gill 
164*cab87e9cSJagpal Singh Gill     while (getline(efile, line))
165*cab87e9cSJagpal Singh Gill     {
166*cab87e9cSJagpal Singh Gill         if (line.substr(0, extendedVersionKey.size())
167*cab87e9cSJagpal Singh Gill                 .find(extendedVersionKey) != std::string::npos)
168*cab87e9cSJagpal Singh Gill         {
169*cab87e9cSJagpal Singh Gill             extendedVersionValue = line.substr(extendedVersionKey.size());
170*cab87e9cSJagpal Singh Gill             std::size_t pos = extendedVersionValue.find_first_of('"') + 1;
171*cab87e9cSJagpal Singh Gill             extendedVersion = extendedVersionValue.substr(
172*cab87e9cSJagpal Singh Gill                 pos, extendedVersionValue.find_last_of('"') - pos);
173*cab87e9cSJagpal Singh Gill             break;
174*cab87e9cSJagpal Singh Gill         }
175*cab87e9cSJagpal Singh Gill     }
176*cab87e9cSJagpal Singh Gill 
177*cab87e9cSJagpal Singh Gill     return extendedVersion;
178*cab87e9cSJagpal Singh Gill }
179*cab87e9cSJagpal Singh Gill 
getBMCVersion(const std::string & releaseFilePath)180*cab87e9cSJagpal Singh Gill std::string Version::getBMCVersion(const std::string& releaseFilePath)
181*cab87e9cSJagpal Singh Gill {
182*cab87e9cSJagpal Singh Gill     std::string versionKey = "VERSION_ID=";
183*cab87e9cSJagpal Singh Gill     std::string versionValue{};
184*cab87e9cSJagpal Singh Gill     std::string version{};
185*cab87e9cSJagpal Singh Gill     std::ifstream efile;
186*cab87e9cSJagpal Singh Gill     std::string line;
187*cab87e9cSJagpal Singh Gill     efile.open(releaseFilePath);
188*cab87e9cSJagpal Singh Gill 
189*cab87e9cSJagpal Singh Gill     while (getline(efile, line))
190*cab87e9cSJagpal Singh Gill     {
191*cab87e9cSJagpal Singh Gill         if (line.substr(0, versionKey.size()).find(versionKey) !=
192*cab87e9cSJagpal Singh Gill             std::string::npos)
193*cab87e9cSJagpal Singh Gill         {
194*cab87e9cSJagpal Singh Gill             // Support quoted and unquoted values
195*cab87e9cSJagpal Singh Gill             // 1. Remove the versionKey so that we process the value only.
196*cab87e9cSJagpal Singh Gill             versionValue = line.substr(versionKey.size());
197*cab87e9cSJagpal Singh Gill 
198*cab87e9cSJagpal Singh Gill             // 2. Look for a starting quote, then increment the position by 1 to
199*cab87e9cSJagpal Singh Gill             //    skip the quote character. If no quote is found,
200*cab87e9cSJagpal Singh Gill             //    find_first_of() returns npos (-1), which by adding +1 sets pos
201*cab87e9cSJagpal Singh Gill             //    to 0 (beginning of unquoted string).
202*cab87e9cSJagpal Singh Gill             std::size_t pos = versionValue.find_first_of('"') + 1;
203*cab87e9cSJagpal Singh Gill 
204*cab87e9cSJagpal Singh Gill             // 3. Look for ending quote, then decrease the position by pos to
205*cab87e9cSJagpal Singh Gill             //    get the size of the string up to before the ending quote. If
206*cab87e9cSJagpal Singh Gill             //    no quote is found, find_last_of() returns npos (-1), and pos
207*cab87e9cSJagpal Singh Gill             //    is 0 for the unquoted case, so substr() is called with a len
208*cab87e9cSJagpal Singh Gill             //    parameter of npos (-1) which according to the documentation
209*cab87e9cSJagpal Singh Gill             //    indicates to use all characters until the end of the string.
210*cab87e9cSJagpal Singh Gill             version =
211*cab87e9cSJagpal Singh Gill                 versionValue.substr(pos, versionValue.find_last_of('"') - pos);
212*cab87e9cSJagpal Singh Gill             break;
213*cab87e9cSJagpal Singh Gill         }
214*cab87e9cSJagpal Singh Gill     }
215*cab87e9cSJagpal Singh Gill     efile.close();
216*cab87e9cSJagpal Singh Gill 
217*cab87e9cSJagpal Singh Gill     if (version.empty())
218*cab87e9cSJagpal Singh Gill     {
219*cab87e9cSJagpal Singh Gill         error("BMC current version is empty");
220*cab87e9cSJagpal Singh Gill         elog<InternalFailure>();
221*cab87e9cSJagpal Singh Gill     }
222*cab87e9cSJagpal Singh Gill 
223*cab87e9cSJagpal Singh Gill     return version;
224*cab87e9cSJagpal Singh Gill }
225*cab87e9cSJagpal Singh Gill 
delete_()226*cab87e9cSJagpal Singh Gill void Delete::delete_()
227*cab87e9cSJagpal Singh Gill {
228*cab87e9cSJagpal Singh Gill     if (parent.eraseCallback)
229*cab87e9cSJagpal Singh Gill     {
230*cab87e9cSJagpal Singh Gill         parent.eraseCallback(parent.id);
231*cab87e9cSJagpal Singh Gill     }
232*cab87e9cSJagpal Singh Gill }
233*cab87e9cSJagpal Singh Gill 
234*cab87e9cSJagpal Singh Gill } // namespace manager
235*cab87e9cSJagpal Singh Gill } // namespace software
236*cab87e9cSJagpal Singh Gill } // namespace phosphor
237