blob: 7a56775d111b39ff34be040375fa470ce3d1a997 [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,
124 filePath)));
Saqib Khan00044f42017-07-10 17:24:43 -0500125 }
Patrick Williams3accb322017-05-30 16:29:52 -0500126 return;
Adriana Kobylak2d8fa222017-03-15 12:34:32 -0500127}
128
Saqib Khan167601b2017-06-18 23:33:46 -0500129void ItemUpdater::processPNORImage()
Saqib Khan7254f0e2017-04-10 21:45:37 -0500130{
Saqib Khan7254f0e2017-04-10 21:45:37 -0500131
Saqib Khan167601b2017-06-18 23:33:46 -0500132 fs::path pnorTOC(PNOR_RO_ACTIVE_PATH);
133 pnorTOC /= PNOR_TOC_FILE;
134 std::ifstream efile(pnorTOC.c_str());
135 if (efile.good() != 1)
Saqib Khan7254f0e2017-04-10 21:45:37 -0500136 {
Saqib Khan167601b2017-06-18 23:33:46 -0500137 log<level::INFO>("Error PNOR current version is empty");
138 return;
Saqib Khan7254f0e2017-04-10 21:45:37 -0500139 }
Saqib Khan167601b2017-06-18 23:33:46 -0500140 auto keyValues = Version::getValue(pnorTOC.string(),
141 std::map<std::string, std::string> {{"version", ""},
142 {"extended_version", ""}});
143 std::string version = keyValues.at("version");
144 std::string extendedVersion = keyValues.at("extended_version");
145 auto id = Version::getId(version);
146 auto purpose = server::Version::VersionPurpose::Host;
147 auto path = std::string{SOFTWARE_OBJPATH} + '/' + id;
148 auto activationState = server::Activation::Activations::Active;
149 activations.insert(std::make_pair(
150 id,
151 std::make_unique<Activation>(
152 bus,
153 path,
154 *this,
155 id,
156 extendedVersion,
157 activationState)));
158 versions.insert(std::make_pair(
159 id,
160 std::make_unique<Version>(
161 bus,
162 path,
163 version,
164 purpose,
165 "")));
166 return;
Saqib Khan7254f0e2017-04-10 21:45:37 -0500167}
168
Adriana Kobylak5ba6b102017-05-19 09:41:27 -0500169int ItemUpdater::validateSquashFSImage(const std::string& filePath)
Saqib Khana8ade7e2017-04-12 10:27:56 -0500170{
Adriana Kobylak5ba6b102017-05-19 09:41:27 -0500171 fs::path file(filePath);
172 file /= squashFSImage;
Saqib Khana8ade7e2017-04-12 10:27:56 -0500173 std::ifstream efile(file.c_str());
174
175 if (efile.good() == 1)
176 {
177 return 0;
178 }
179 else
180 {
181 log<level::ERR>("Failed to find the SquashFS image.");
182 return -1;
183 }
184}
185
Michael Tritzdd961b62017-05-17 14:07:03 -0500186void ItemUpdater::reset()
187{
188 for(const auto& it : activations)
189 {
190 auto serviceFile = "obmc-flash-bios-ubiumount-rw@" + it.first +
191 ".service";
192
193 // Remove the read-write partitions.
Adriana Kobylakd6a549e2017-05-10 16:23:01 -0500194 auto method = bus.new_method_call(
Michael Tritzdd961b62017-05-17 14:07:03 -0500195 SYSTEMD_BUSNAME,
196 SYSTEMD_PATH,
197 SYSTEMD_INTERFACE,
198 "StartUnit");
199 method.append(serviceFile, "replace");
Adriana Kobylakd6a549e2017-05-10 16:23:01 -0500200 bus.call_noreply(method);
Michael Tritzdd961b62017-05-17 14:07:03 -0500201 }
202
203 // Remove the preserved partition.
Adriana Kobylakd6a549e2017-05-10 16:23:01 -0500204 auto method = bus.new_method_call(
Michael Tritzdd961b62017-05-17 14:07:03 -0500205 SYSTEMD_BUSNAME,
206 SYSTEMD_PATH,
207 SYSTEMD_INTERFACE,
208 "StartUnit");
209 method.append("obmc-flash-bios-ubiumount-prsv.service", "replace");
Adriana Kobylakd6a549e2017-05-10 16:23:01 -0500210 bus.call_noreply(method);
Michael Tritzdd961b62017-05-17 14:07:03 -0500211
212 return;
213}
214
Saqib Khan81bac882017-06-08 12:17:01 -0500215void ItemUpdater::freePriority(uint8_t value)
216{
217 //TODO openbmc/openbmc#1896 Improve the performance of this function
218 for (const auto& intf : activations)
219 {
220 if(intf.second->redundancyPriority)
221 {
222 if (intf.second->redundancyPriority.get()->priority() == value)
223 {
224 intf.second->redundancyPriority.get()->priority(value+1);
225 }
226 }
227 }
228}
229
Adriana Kobylakbefe5ce2017-04-05 15:57:44 -0500230} // namespace updater
Adriana Kobylak2d8fa222017-03-15 12:34:32 -0500231} // namespace software
232} // namespace openpower