blob: 1c0a6b866bc54199de160e97689d2b240d0f1649 [file] [log] [blame]
Saqib Khan7254f0e2017-04-10 21:45:37 -05001#include <string>
Adriana Kobylak5ba6b102017-05-19 09:41:27 -05002#include <experimental/filesystem>
Saqib Khan7254f0e2017-04-10 21:45:37 -05003#include <fstream>
Adriana Kobylakb66ac3a2017-03-28 13:33:20 -05004#include <phosphor-logging/log.hpp>
Adriana Kobylakd6a549e2017-05-10 16:23:01 -05005#include <xyz/openbmc_project/Software/Version/server.hpp>
Saqib Khan167601b2017-06-18 23:33:46 -05006#include "version.hpp"
Adriana Kobylak2d8fa222017-03-15 12:34:32 -05007#include "config.h"
8#include "item_updater.hpp"
Saqib Khana8ade7e2017-04-12 10:27:56 -05009#include "activation.hpp"
Adriana Kobylak2d8fa222017-03-15 12:34:32 -050010
11namespace openpower
12{
13namespace software
14{
Adriana Kobylakbefe5ce2017-04-05 15:57:44 -050015namespace updater
Adriana Kobylak2d8fa222017-03-15 12:34:32 -050016{
17
Saqib Khana8ade7e2017-04-12 10:27:56 -050018// When you see server:: you know we're referencing our base class
19namespace server = sdbusplus::xyz::openbmc_project::Software::server;
Adriana Kobylak5ba6b102017-05-19 09:41:27 -050020namespace fs = std::experimental::filesystem;
Saqib Khana8ade7e2017-04-12 10:27:56 -050021
Adriana Kobylakb66ac3a2017-03-28 13:33:20 -050022using namespace phosphor::logging;
23
Saqib Khana8ade7e2017-04-12 10:27:56 -050024constexpr auto squashFSImage = "pnor.xz.squashfs";
25
Adriana Kobylakd6a549e2017-05-10 16:23:01 -050026void ItemUpdater::createActivation(sdbusplus::message::message& m)
Adriana Kobylak2d8fa222017-03-15 12:34:32 -050027{
Patrick Williamse4290942017-06-16 05:43:08 -050028 using SVersion = server::Version;
29 using VersionPurpose = SVersion::VersionPurpose;
30 namespace msg = sdbusplus::message;
31 namespace variant_ns = msg::variant_ns;
32
Adriana Kobylakd6a549e2017-05-10 16:23:01 -050033 sdbusplus::message::object_path objPath;
34 std::map<std::string,
Patrick Williamse4290942017-06-16 05:43:08 -050035 std::map<std::string, msg::variant<std::string>>> interfaces;
Adriana Kobylakd6a549e2017-05-10 16:23:01 -050036 m.read(objPath, interfaces);
Patrick Williamse4290942017-06-16 05:43:08 -050037
Adriana Kobylakd6a549e2017-05-10 16:23:01 -050038 std::string path(std::move(objPath));
Adriana Kobylak5ba6b102017-05-19 09:41:27 -050039 std::string filePath;
Patrick Williamse4290942017-06-16 05:43:08 -050040 auto purpose = VersionPurpose::Unknown;
Saqib Khance148702017-06-11 12:01:58 -050041 std::string version;
Adriana Kobylak5ba6b102017-05-19 09:41:27 -050042
Adriana Kobylakd6a549e2017-05-10 16:23:01 -050043 for (const auto& intf : interfaces)
Adriana Kobylakb66ac3a2017-03-28 13:33:20 -050044 {
Adriana Kobylakd6a549e2017-05-10 16:23:01 -050045 if (intf.first == VERSION_IFACE)
46 {
47 for (const auto& property : intf.second)
48 {
49 if (property.first == "Purpose")
50 {
51 // Only process the Host and System images
Patrick Williamse4290942017-06-16 05:43:08 -050052 auto value = SVersion::convertVersionPurposeFromString(
53 variant_ns::get<std::string>(property.second));
54
55 if (value == VersionPurpose::Host ||
56 value == VersionPurpose::System)
Adriana Kobylakd6a549e2017-05-10 16:23:01 -050057 {
Saqib Khance148702017-06-11 12:01:58 -050058 purpose = value;
Adriana Kobylakd6a549e2017-05-10 16:23:01 -050059 }
60 }
Saqib Khance148702017-06-11 12:01:58 -050061 else if (property.first == "Version")
62 {
Patrick Williamse4290942017-06-16 05:43:08 -050063 version = variant_ns::get<std::string>(property.second);
Saqib Khance148702017-06-11 12:01:58 -050064 }
Adriana Kobylakd6a549e2017-05-10 16:23:01 -050065 }
66 }
Adriana Kobylak5ba6b102017-05-19 09:41:27 -050067 else if (intf.first == FILEPATH_IFACE)
68 {
69 for (const auto& property : intf.second)
70 {
71 if (property.first == "Path")
72 {
Patrick Williamse4290942017-06-16 05:43:08 -050073 filePath = variant_ns::get<std::string>(property.second);
Adriana Kobylak5ba6b102017-05-19 09:41:27 -050074 }
75 }
76 }
77 }
Patrick Williamse4290942017-06-16 05:43:08 -050078 if ((filePath.empty()) || (purpose == VersionPurpose::Unknown))
Adriana Kobylak5ba6b102017-05-19 09:41:27 -050079 {
80 return;
Adriana Kobylakb66ac3a2017-03-28 13:33:20 -050081 }
82
Adriana Kobylakd6a549e2017-05-10 16:23:01 -050083 // Version id is the last item in the path
84 auto pos = path.rfind("/");
85 if (pos == std::string::npos)
Adriana Kobylakb66ac3a2017-03-28 13:33:20 -050086 {
Adriana Kobylakd6a549e2017-05-10 16:23:01 -050087 log<level::ERR>("No version id found in object path",
88 entry("OBJPATH=%s", path));
89 return;
90 }
91
92 auto versionId = path.substr(pos + 1);
93
94 if (activations.find(versionId) == activations.end())
95 {
96 // Determine the Activation state by processing the given image dir.
97 auto activationState = server::Activation::Activations::Invalid;
Adriana Kobylak5ba6b102017-05-19 09:41:27 -050098 if (ItemUpdater::validateSquashFSImage(filePath) == 0)
Adriana Kobylakb66ac3a2017-03-28 13:33:20 -050099 {
Adriana Kobylakd6a549e2017-05-10 16:23:01 -0500100 activationState = server::Activation::Activations::Ready;
Adriana Kobylakb66ac3a2017-03-28 13:33:20 -0500101 }
102
Adriana Kobylak5ba6b102017-05-19 09:41:27 -0500103 fs::path manifestPath(filePath);
104 manifestPath /= MANIFEST_FILE;
Saqib Khan167601b2017-06-18 23:33:46 -0500105 std::string extendedVersion = (Version::getValue(manifestPath.string(),
106 std::map<std::string, std::string>
107 {{"extended_version", ""}})).begin()->second;
Adriana Kobylakd6a549e2017-05-10 16:23:01 -0500108 activations.insert(std::make_pair(
109 versionId,
110 std::make_unique<Activation>(
111 bus,
112 path,
Saqib Khan81bac882017-06-08 12:17:01 -0500113 *this,
Adriana Kobylakd6a549e2017-05-10 16:23:01 -0500114 versionId,
115 extendedVersion,
116 activationState)));
Saqib Khance148702017-06-11 12:01:58 -0500117 versions.insert(std::make_pair(
118 versionId,
119 std::make_unique<Version>(
120 bus,
121 path,
122 version,
123 purpose,
Leonel Gonzalez9c8adfa2017-07-12 11:08:40 -0500124 filePath,
125 *this)));
Saqib Khan00044f42017-07-10 17:24:43 -0500126 }
Patrick Williams3accb322017-05-30 16:29:52 -0500127 return;
Adriana Kobylak2d8fa222017-03-15 12:34:32 -0500128}
129
Saqib Khan167601b2017-06-18 23:33:46 -0500130void ItemUpdater::processPNORImage()
Saqib Khan7254f0e2017-04-10 21:45:37 -0500131{
Saqib Khan7254f0e2017-04-10 21:45:37 -0500132
Saqib Khan167601b2017-06-18 23:33:46 -0500133 fs::path pnorTOC(PNOR_RO_ACTIVE_PATH);
134 pnorTOC /= PNOR_TOC_FILE;
135 std::ifstream efile(pnorTOC.c_str());
136 if (efile.good() != 1)
Saqib Khan7254f0e2017-04-10 21:45:37 -0500137 {
Saqib Khan167601b2017-06-18 23:33:46 -0500138 log<level::INFO>("Error PNOR current version is empty");
139 return;
Saqib Khan7254f0e2017-04-10 21:45:37 -0500140 }
Saqib Khan167601b2017-06-18 23:33:46 -0500141 auto keyValues = Version::getValue(pnorTOC.string(),
142 std::map<std::string, std::string> {{"version", ""},
143 {"extended_version", ""}});
144 std::string version = keyValues.at("version");
145 std::string extendedVersion = keyValues.at("extended_version");
146 auto id = Version::getId(version);
147 auto purpose = server::Version::VersionPurpose::Host;
148 auto path = std::string{SOFTWARE_OBJPATH} + '/' + id;
149 auto activationState = server::Activation::Activations::Active;
150 activations.insert(std::make_pair(
151 id,
152 std::make_unique<Activation>(
153 bus,
154 path,
155 *this,
156 id,
157 extendedVersion,
158 activationState)));
159 versions.insert(std::make_pair(
160 id,
161 std::make_unique<Version>(
162 bus,
163 path,
164 version,
165 purpose,
Leonel Gonzalez9c8adfa2017-07-12 11:08:40 -0500166 "",
167 *this)));
Saqib Khan167601b2017-06-18 23:33:46 -0500168 return;
Saqib Khan7254f0e2017-04-10 21:45:37 -0500169}
170
Adriana Kobylak5ba6b102017-05-19 09:41:27 -0500171int ItemUpdater::validateSquashFSImage(const std::string& filePath)
Saqib Khana8ade7e2017-04-12 10:27:56 -0500172{
Adriana Kobylak5ba6b102017-05-19 09:41:27 -0500173 fs::path file(filePath);
174 file /= squashFSImage;
Saqib Khana8ade7e2017-04-12 10:27:56 -0500175 std::ifstream efile(file.c_str());
176
177 if (efile.good() == 1)
178 {
179 return 0;
180 }
181 else
182 {
183 log<level::ERR>("Failed to find the SquashFS image.");
184 return -1;
185 }
186}
187
Leonel Gonzalez9c8adfa2017-07-12 11:08:40 -0500188void ItemUpdater::removeReadOnlyPartition(std::string versionId)
Michael Tritzdd961b62017-05-17 14:07:03 -0500189{
Leonel Gonzalez9c8adfa2017-07-12 11:08:40 -0500190 auto serviceFile = "obmc-flash-bios-ubiumount-ro@" + versionId +
191 ".service";
192
193 // Remove the read-only partitions.
194 auto method = bus.new_method_call(
195 SYSTEMD_BUSNAME,
196 SYSTEMD_PATH,
197 SYSTEMD_INTERFACE,
198 "StartUnit");
199 method.append(serviceFile, "replace");
200 bus.call_noreply(method);
201}
202
203void ItemUpdater::removeReadWritePartition(std::string versionId)
204{
205 auto serviceFile = "obmc-flash-bios-ubiumount-rw@" + versionId +
Michael Tritzdd961b62017-05-17 14:07:03 -0500206 ".service";
207
208 // Remove the read-write partitions.
Adriana Kobylakd6a549e2017-05-10 16:23:01 -0500209 auto method = bus.new_method_call(
Michael Tritzdd961b62017-05-17 14:07:03 -0500210 SYSTEMD_BUSNAME,
211 SYSTEMD_PATH,
212 SYSTEMD_INTERFACE,
213 "StartUnit");
214 method.append(serviceFile, "replace");
Adriana Kobylakd6a549e2017-05-10 16:23:01 -0500215 bus.call_noreply(method);
Leonel Gonzalez9c8adfa2017-07-12 11:08:40 -0500216}
Michael Tritzdd961b62017-05-17 14:07:03 -0500217
Leonel Gonzalez9c8adfa2017-07-12 11:08:40 -0500218void ItemUpdater::removePreservedPartition()
219{
Michael Tritzdd961b62017-05-17 14:07:03 -0500220 // Remove the preserved partition.
Adriana Kobylakd6a549e2017-05-10 16:23:01 -0500221 auto method = bus.new_method_call(
Michael Tritzdd961b62017-05-17 14:07:03 -0500222 SYSTEMD_BUSNAME,
223 SYSTEMD_PATH,
224 SYSTEMD_INTERFACE,
225 "StartUnit");
226 method.append("obmc-flash-bios-ubiumount-prsv.service", "replace");
Adriana Kobylakd6a549e2017-05-10 16:23:01 -0500227 bus.call_noreply(method);
Michael Tritzdd961b62017-05-17 14:07:03 -0500228
229 return;
230}
231
Leonel Gonzalez9c8adfa2017-07-12 11:08:40 -0500232void ItemUpdater::reset()
233{
234 for(const auto& it : activations)
235 {
236 removeReadWritePartition(it.first);
237 }
238 removePreservedPartition();
239 return;
240}
241
Saqib Khan81bac882017-06-08 12:17:01 -0500242void ItemUpdater::freePriority(uint8_t value)
243{
244 //TODO openbmc/openbmc#1896 Improve the performance of this function
245 for (const auto& intf : activations)
246 {
247 if(intf.second->redundancyPriority)
248 {
249 if (intf.second->redundancyPriority.get()->priority() == value)
250 {
251 intf.second->redundancyPriority.get()->priority(value+1);
252 }
253 }
254 }
255}
256
Saqib Khan2af5c492017-07-17 16:15:13 -0500257bool ItemUpdater::isLowestPriority(uint8_t value)
258{
259 for (const auto& intf : activations)
260 {
261 if(intf.second->redundancyPriority)
262 {
263 if (intf.second->redundancyPriority.get()->priority() < value)
264 {
265 return false;
266 }
267 }
268 }
269 return true;
270}
271
Leonel Gonzalez9c8adfa2017-07-12 11:08:40 -0500272void ItemUpdater::erase(std::string entryId)
273{
274 // Removing partitions
275 removeReadWritePartition(entryId);
276 removeReadOnlyPartition(entryId);
277
278 // Removing entry in versions map
279 auto it = versions.find(entryId);
280 if (it == versions.end())
281 {
282 log<level::ERR>(("Error: Failed to find version " + entryId + \
283 " in item updater versions map." \
284 " Unable to remove.").c_str());
285 return;
286 }
287 versions.erase(entryId);
288
289 // Removing entry in activations map
290 auto ita = activations.find(entryId);
291 if (ita == activations.end())
292 {
293 log<level::ERR>(("Error: Failed to find version " + entryId + \
294 " in item updater activations map." \
295 " Unable to remove.").c_str());
296 return;
297 }
298 activations.erase(entryId);
299}
300
Adriana Kobylakbefe5ce2017-04-05 15:57:44 -0500301} // namespace updater
Adriana Kobylak2d8fa222017-03-15 12:34:32 -0500302} // namespace software
303} // namespace openpower