xref: /openbmc/phosphor-bmc-code-mgmt/item_updater.cpp (revision c026f6cdc3f7e305d98dc824fcba0f199a7aa157)
1 #include "config.h"
2 
3 #include "item_updater.hpp"
4 
5 #include "images.hpp"
6 #include "serialize.hpp"
7 #include "version.hpp"
8 #include "xyz/openbmc_project/Software/ExtendedVersion/server.hpp"
9 #include "xyz/openbmc_project/Software/Version/server.hpp"
10 
11 #include <phosphor-logging/elog-errors.hpp>
12 #include <phosphor-logging/elog.hpp>
13 #include <phosphor-logging/lg2.hpp>
14 #include <xyz/openbmc_project/Common/error.hpp>
15 #include <xyz/openbmc_project/Software/Image/error.hpp>
16 
17 #include <filesystem>
18 #include <fstream>
19 #include <queue>
20 #include <set>
21 #include <string>
22 
23 namespace phosphor
24 {
25 namespace software
26 {
27 namespace updater
28 {
29 
30 // When you see server:: you know we're referencing our base class
31 namespace server = sdbusplus::xyz::openbmc_project::Software::server;
32 namespace control = sdbusplus::xyz::openbmc_project::Control::server;
33 
34 PHOSPHOR_LOG2_USING;
35 using namespace phosphor::logging;
36 using namespace sdbusplus::xyz::openbmc_project::Software::Image::Error;
37 using namespace phosphor::software::image;
38 namespace fs = std::filesystem;
39 using NotAllowed = sdbusplus::xyz::openbmc_project::Common::Error::NotAllowed;
40 
41 void ItemUpdater::createActivation(sdbusplus::message::message& msg)
42 {
43 
44     using SVersion = server::Version;
45     using VersionPurpose = SVersion::VersionPurpose;
46     using VersionClass = phosphor::software::manager::Version;
47 
48     sdbusplus::message::object_path objPath;
49     auto purpose = VersionPurpose::Unknown;
50     std::string extendedVersion;
51     std::string version;
52     std::map<std::string, std::map<std::string, std::variant<std::string>>>
53         interfaces;
54     msg.read(objPath, interfaces);
55     std::string path(std::move(objPath));
56     std::string filePath;
57 
58     for (const auto& intf : interfaces)
59     {
60         if (intf.first == VERSION_IFACE)
61         {
62             for (const auto& property : intf.second)
63             {
64                 if (property.first == "Purpose")
65                 {
66                     auto value = SVersion::convertVersionPurposeFromString(
67                         std::get<std::string>(property.second));
68                     if (value == VersionPurpose::BMC ||
69 #ifdef HOST_BIOS_UPGRADE
70                         value == VersionPurpose::Host ||
71 #endif
72                         value == VersionPurpose::System)
73                     {
74                         purpose = value;
75                     }
76                 }
77                 else if (property.first == "Version")
78                 {
79                     version = std::get<std::string>(property.second);
80                 }
81             }
82         }
83         else if (intf.first == FILEPATH_IFACE)
84         {
85             for (const auto& property : intf.second)
86             {
87                 if (property.first == "Path")
88                 {
89                     filePath = std::get<std::string>(property.second);
90                 }
91             }
92         }
93         else if (intf.first == EXTENDED_VERSION_IFACE)
94         {
95             for (const auto& property : intf.second)
96             {
97                 if (property.first == "ExtendedVersion")
98                 {
99                     extendedVersion = std::get<std::string>(property.second);
100                 }
101             }
102         }
103     }
104     if (version.empty() || filePath.empty() ||
105         purpose == VersionPurpose::Unknown)
106     {
107         return;
108     }
109 
110     // Version id is the last item in the path
111     auto pos = path.rfind("/");
112     if (pos == std::string::npos)
113     {
114         error("No version id found in object path: {PATH}", "PATH", path);
115         return;
116     }
117 
118     auto versionId = path.substr(pos + 1);
119 
120     if (activations.find(versionId) == activations.end())
121     {
122         // Determine the Activation state by processing the given image dir.
123         auto activationState = server::Activation::Activations::Invalid;
124         ItemUpdater::ActivationStatus result;
125         if (purpose == VersionPurpose::BMC || purpose == VersionPurpose::System)
126             result = ItemUpdater::validateSquashFSImage(filePath);
127         else
128             result = ItemUpdater::ActivationStatus::ready;
129 
130         AssociationList associations = {};
131 
132         if (result == ItemUpdater::ActivationStatus::ready)
133         {
134             activationState = server::Activation::Activations::Ready;
135             // Create an association to the BMC inventory item
136             associations.emplace_back(
137                 std::make_tuple(ACTIVATION_FWD_ASSOCIATION,
138                                 ACTIVATION_REV_ASSOCIATION, bmcInventoryPath));
139         }
140 
141         auto versionPtr = std::make_unique<VersionClass>(
142             bus, path, version, purpose, extendedVersion, filePath,
143             std::bind(&ItemUpdater::erase, this, std::placeholders::_1),
144             versionId);
145         versionPtr->deleteObject =
146             std::make_unique<phosphor::software::manager::Delete>(bus, path,
147                                                                   *versionPtr);
148         versions.insert(std::make_pair(versionId, std::move(versionPtr)));
149 
150         activations.insert(std::make_pair(
151             versionId,
152             std::make_unique<Activation>(bus, path, *this, versionId,
153                                          activationState, associations)));
154     }
155     return;
156 }
157 
158 void ItemUpdater::processBMCImage()
159 {
160     using VersionClass = phosphor::software::manager::Version;
161 
162     // Check MEDIA_DIR and create if it does not exist
163     try
164     {
165         if (!fs::is_directory(MEDIA_DIR))
166         {
167             fs::create_directory(MEDIA_DIR);
168         }
169     }
170     catch (const fs::filesystem_error& e)
171     {
172         error("Failed to prepare dir: {ERROR}", "ERROR", e);
173         return;
174     }
175 
176     // Functional images are mounted as rofs-<location>-functional
177     constexpr auto functionalSuffix = "-functional";
178     bool functionalFound = false;
179 
180     // Read os-release from folders under /media/ to get
181     // BMC Software Versions.
182     for (const auto& iter : fs::directory_iterator(MEDIA_DIR))
183     {
184         auto activationState = server::Activation::Activations::Active;
185         static const auto BMC_RO_PREFIX_LEN = strlen(BMC_ROFS_PREFIX);
186 
187         // Check if the BMC_RO_PREFIXis the prefix of the iter.path
188         if (0 ==
189             iter.path().native().compare(0, BMC_RO_PREFIX_LEN, BMC_ROFS_PREFIX))
190         {
191             // Get the version to calculate the id
192             fs::path releaseFile(OS_RELEASE_FILE);
193             auto osRelease = iter.path() / releaseFile.relative_path();
194             if (!fs::is_regular_file(osRelease))
195             {
196 #ifdef BMC_STATIC_DUAL_IMAGE
197                 // For dual image, it is possible that the secondary image is
198                 // empty or contains invalid data, ignore such case.
199                 info("Unable to find osRelease: {PATH}", "PATH", osRelease);
200 #else
201                 error("Failed to read osRelease: {PATH}", "PATH", osRelease);
202 
203                 // Try to get the version id from the mount directory name and
204                 // call to delete it as this version may be corrupted. Dynamic
205                 // volumes created by the UBI layout for example have the id in
206                 // the mount directory name. The worst that can happen is that
207                 // erase() is called with an non-existent id and returns.
208                 auto id = iter.path().native().substr(BMC_RO_PREFIX_LEN);
209                 ItemUpdater::erase(id);
210 #endif
211 
212                 continue;
213             }
214             auto version = VersionClass::getBMCVersion(osRelease);
215             if (version.empty())
216             {
217                 error("Failed to read version from osRelease: {PATH}", "PATH",
218                       osRelease);
219 
220                 // Try to delete the version, same as above if the
221                 // OS_RELEASE_FILE does not exist.
222                 auto id = iter.path().native().substr(BMC_RO_PREFIX_LEN);
223                 ItemUpdater::erase(id);
224 
225                 continue;
226             }
227 
228             // The flash location is part of the mount name: rofs-<location>
229             auto flashId = iter.path().native().substr(BMC_RO_PREFIX_LEN);
230 
231             auto id = VersionClass::getId(version + flashId);
232 
233             // Check if the id has already been added. This can happen if the
234             // BMC partitions / devices were manually flashed with the same
235             // image.
236             if (versions.find(id) != versions.end())
237             {
238                 continue;
239             }
240 
241             auto functional = false;
242             if (iter.path().native().find(functionalSuffix) !=
243                 std::string::npos)
244             {
245                 // Set functional to true and remove the functional suffix
246                 functional = true;
247                 flashId.erase(flashId.length() - strlen(functionalSuffix));
248                 functionalFound = true;
249             }
250 
251             auto purpose = server::Version::VersionPurpose::BMC;
252             restorePurpose(flashId, purpose);
253 
254             // Read os-release from /etc/ to get the BMC extended version
255             std::string extendedVersion =
256                 VersionClass::getBMCExtendedVersion(osRelease);
257 
258             auto path = fs::path(SOFTWARE_OBJPATH) / id;
259 
260             // Create functional association if this is the functional
261             // version
262             if (functional)
263             {
264                 createFunctionalAssociation(path);
265             }
266 
267             AssociationList associations = {};
268 
269             if (activationState == server::Activation::Activations::Active)
270             {
271                 // Create an association to the BMC inventory item
272                 associations.emplace_back(std::make_tuple(
273                     ACTIVATION_FWD_ASSOCIATION, ACTIVATION_REV_ASSOCIATION,
274                     bmcInventoryPath));
275 
276                 // Create an active association since this image is active
277                 createActiveAssociation(path);
278             }
279 
280             // All updateable firmware components must expose the updateable
281             // association.
282             createUpdateableAssociation(path);
283 
284             // Create Version instance for this version.
285             auto versionPtr = std::make_unique<VersionClass>(
286                 bus, path, version, purpose, extendedVersion, flashId,
287                 std::bind(&ItemUpdater::erase, this, std::placeholders::_1),
288                 id);
289             if (functional)
290             {
291                 versionPtr->setFunctional(true);
292             }
293             else
294             {
295                 versionPtr->deleteObject =
296                     std::make_unique<phosphor::software::manager::Delete>(
297                         bus, path, *versionPtr);
298             }
299             versions.insert(std::make_pair(id, std::move(versionPtr)));
300 
301             // Create Activation instance for this version.
302             activations.insert(std::make_pair(
303                 id, std::make_unique<Activation>(
304                         bus, path, *this, id, activationState, associations)));
305 
306 #ifdef BMC_STATIC_DUAL_IMAGE
307             uint8_t priority;
308             if ((functional && (runningImageSlot == 0)) ||
309                 (!functional && (runningImageSlot == 1)))
310             {
311                 priority = 0;
312             }
313             else
314             {
315                 priority = 1;
316             }
317             activations.find(id)->second->redundancyPriority =
318                 std::make_unique<RedundancyPriority>(
319                     bus, path, *(activations.find(id)->second), priority,
320                     false);
321 #else
322             // If Active, create RedundancyPriority instance for this
323             // version.
324             if (activationState == server::Activation::Activations::Active)
325             {
326                 uint8_t priority = std::numeric_limits<uint8_t>::max();
327                 if (!restorePriority(flashId, priority))
328                 {
329                     if (functional)
330                     {
331                         priority = 0;
332                     }
333                     else
334                     {
335                         error(
336                             "Unable to restore priority from file for {VERSIONID}",
337                             "VERSIONID", id);
338                     }
339                 }
340                 activations.find(id)->second->redundancyPriority =
341                     std::make_unique<RedundancyPriority>(
342                         bus, path, *(activations.find(id)->second), priority,
343                         false);
344             }
345 #endif
346         }
347     }
348 
349     if (!functionalFound)
350     {
351         // If there is no functional version found, read the /etc/os-release and
352         // create rofs-<versionId>-functional under MEDIA_DIR, then call again
353         // processBMCImage() to create the D-Bus interface for it.
354         auto version = VersionClass::getBMCVersion(OS_RELEASE_FILE);
355         auto id = phosphor::software::manager::Version::getId(version +
356                                                               functionalSuffix);
357         auto versionFileDir = BMC_ROFS_PREFIX + id + functionalSuffix + "/etc/";
358         try
359         {
360             if (!fs::is_directory(versionFileDir))
361             {
362                 fs::create_directories(versionFileDir);
363             }
364             auto versionFilePath =
365                 BMC_ROFS_PREFIX + id + functionalSuffix + OS_RELEASE_FILE;
366             fs::create_directory_symlink(OS_RELEASE_FILE, versionFilePath);
367             ItemUpdater::processBMCImage();
368         }
369         catch (const std::exception& e)
370         {
371             error("Exception during processing: {ERROR}", "ERROR", e);
372         }
373     }
374 
375     mirrorUbootToAlt();
376     return;
377 }
378 
379 void ItemUpdater::erase(std::string entryId)
380 {
381     // Find entry in versions map
382     auto it = versions.find(entryId);
383     if (it != versions.end())
384     {
385         if (it->second->isFunctional() && ACTIVE_BMC_MAX_ALLOWED > 1)
386         {
387             error(
388                 "Version ({VERSIONID}) is currently running on the BMC; unable to remove.",
389                 "VERSIONID", entryId);
390             return;
391         }
392     }
393 
394     // First call resetUbootEnvVars() so that the BMC points to a valid image to
395     // boot from. If resetUbootEnvVars() is called after the image is actually
396     // deleted from the BMC flash, there'd be a time window where the BMC would
397     // be pointing to a non-existent image to boot from.
398     // Need to remove the entries from the activations map before that call so
399     // that resetUbootEnvVars() doesn't use the version to be deleted.
400     auto iteratorActivations = activations.find(entryId);
401     if (iteratorActivations == activations.end())
402     {
403         error(
404             "Failed to find version ({VERSIONID}) in item updater activations map; unable to remove.",
405             "VERSIONID", entryId);
406     }
407     else
408     {
409         removeAssociations(iteratorActivations->second->path);
410         iteratorActivations->second->deleteImageManagerObject();
411         this->activations.erase(entryId);
412     }
413     ItemUpdater::resetUbootEnvVars();
414 
415     if (it != versions.end())
416     {
417         auto flashId = it->second->path();
418 
419         // Delete version data if it has been installed on flash (path is not
420         // the upload directory)
421         if (flashId.find(IMG_UPLOAD_DIR) == std::string::npos)
422         {
423             removeReadOnlyPartition(entryId);
424             removePersistDataDirectory(flashId);
425             helper.clearEntry(flashId);
426         }
427 
428         // Removing entry in versions map
429         this->versions.erase(entryId);
430     }
431 
432     return;
433 }
434 
435 void ItemUpdater::deleteAll()
436 {
437     std::vector<std::string> deletableVersions;
438 
439     for (const auto& versionIt : versions)
440     {
441         if (!versionIt.second->isFunctional())
442         {
443             deletableVersions.push_back(versionIt.first);
444         }
445     }
446 
447     for (const auto& deletableIt : deletableVersions)
448     {
449         ItemUpdater::erase(deletableIt);
450     }
451 
452     helper.cleanup();
453 }
454 
455 ItemUpdater::ActivationStatus
456     ItemUpdater::validateSquashFSImage(const std::string& filePath)
457 {
458     bool valid = true;
459 
460     // Record the images which are being updated
461     // First check for the fullimage, then check for images with partitions
462     imageUpdateList.push_back(bmcFullImages);
463     valid = checkImage(filePath, imageUpdateList);
464     if (!valid)
465     {
466         imageUpdateList.clear();
467         imageUpdateList.assign(bmcImages.begin(), bmcImages.end());
468         valid = checkImage(filePath, imageUpdateList);
469         if (!valid)
470         {
471             error("Failed to find the needed BMC images.");
472             return ItemUpdater::ActivationStatus::invalid;
473         }
474     }
475 
476     return ItemUpdater::ActivationStatus::ready;
477 }
478 
479 void ItemUpdater::savePriority(const std::string& versionId, uint8_t value)
480 {
481     auto flashId = versions.find(versionId)->second->path();
482     storePriority(flashId, value);
483     helper.setEntry(flashId, value);
484 }
485 
486 void ItemUpdater::freePriority(uint8_t value, const std::string& versionId)
487 {
488     std::map<std::string, uint8_t> priorityMap;
489 
490     // Insert the requested version and priority, it may not exist yet.
491     priorityMap.insert(std::make_pair(versionId, value));
492 
493     for (const auto& intf : activations)
494     {
495         if (intf.second->redundancyPriority)
496         {
497             priorityMap.insert(std::make_pair(
498                 intf.first, intf.second->redundancyPriority.get()->priority()));
499         }
500     }
501 
502     // Lambda function to compare 2 priority values, use <= to allow duplicates
503     typedef std::function<bool(std::pair<std::string, uint8_t>,
504                                std::pair<std::string, uint8_t>)>
505         cmpPriority;
506     cmpPriority cmpPriorityFunc =
507         [](std::pair<std::string, uint8_t> priority1,
508            std::pair<std::string, uint8_t> priority2) {
509             return priority1.second <= priority2.second;
510         };
511 
512     // Sort versions by ascending priority
513     std::set<std::pair<std::string, uint8_t>, cmpPriority> prioritySet(
514         priorityMap.begin(), priorityMap.end(), cmpPriorityFunc);
515 
516     auto freePriorityValue = value;
517     for (auto& element : prioritySet)
518     {
519         if (element.first == versionId)
520         {
521             continue;
522         }
523         if (element.second == freePriorityValue)
524         {
525             ++freePriorityValue;
526             auto it = activations.find(element.first);
527             it->second->redundancyPriority.get()->sdbusPriority(
528                 freePriorityValue);
529         }
530     }
531 
532     auto lowestVersion = prioritySet.begin()->first;
533     if (value == prioritySet.begin()->second)
534     {
535         lowestVersion = versionId;
536     }
537     updateUbootEnvVars(lowestVersion);
538 }
539 
540 void ItemUpdater::reset()
541 {
542     helper.factoryReset();
543 
544     info("BMC factory reset will take effect upon reboot.");
545 }
546 
547 void ItemUpdater::removeReadOnlyPartition(std::string versionId)
548 {
549     auto flashId = versions.find(versionId)->second->path();
550     helper.removeVersion(flashId);
551 }
552 
553 bool ItemUpdater::fieldModeEnabled(bool value)
554 {
555     // enabling field mode is intended to be one way: false -> true
556     if (value && !control::FieldMode::fieldModeEnabled())
557     {
558         control::FieldMode::fieldModeEnabled(value);
559 
560         auto method = bus.new_method_call(SYSTEMD_BUSNAME, SYSTEMD_PATH,
561                                           SYSTEMD_INTERFACE, "StartUnit");
562         method.append("obmc-flash-bmc-setenv@fieldmode\\x3dtrue.service",
563                       "replace");
564         bus.call_noreply(method);
565 
566         method = bus.new_method_call(SYSTEMD_BUSNAME, SYSTEMD_PATH,
567                                      SYSTEMD_INTERFACE, "StopUnit");
568         method.append("usr-local.mount", "replace");
569         bus.call_noreply(method);
570 
571         std::vector<std::string> usrLocal = {"usr-local.mount"};
572 
573         method = bus.new_method_call(SYSTEMD_BUSNAME, SYSTEMD_PATH,
574                                      SYSTEMD_INTERFACE, "MaskUnitFiles");
575         method.append(usrLocal, false, true);
576         bus.call_noreply(method);
577     }
578     else if (!value && control::FieldMode::fieldModeEnabled())
579     {
580         elog<NotAllowed>(xyz::openbmc_project::Common::NotAllowed::REASON(
581             "FieldMode is not allowed to be cleared"));
582     }
583 
584     return control::FieldMode::fieldModeEnabled();
585 }
586 
587 void ItemUpdater::restoreFieldModeStatus()
588 {
589     std::ifstream input("/dev/mtd/u-boot-env");
590     std::string envVar;
591     std::getline(input, envVar);
592 
593     if (envVar.find("fieldmode=true") != std::string::npos)
594     {
595         ItemUpdater::fieldModeEnabled(true);
596     }
597 }
598 
599 void ItemUpdater::setBMCInventoryPath()
600 {
601     auto depth = 0;
602     auto mapperCall = bus.new_method_call(MAPPER_BUSNAME, MAPPER_PATH,
603                                           MAPPER_INTERFACE, "GetSubTreePaths");
604 
605     mapperCall.append(INVENTORY_PATH);
606     mapperCall.append(depth);
607     std::vector<std::string> filter = {BMC_INVENTORY_INTERFACE};
608     mapperCall.append(filter);
609 
610     try
611     {
612         auto response = bus.call(mapperCall);
613 
614         using ObjectPaths = std::vector<std::string>;
615         ObjectPaths result;
616         response.read(result);
617 
618         if (!result.empty())
619         {
620             bmcInventoryPath = result.front();
621         }
622     }
623     catch (const sdbusplus::exception::exception& e)
624     {
625         error("Error in mapper GetSubTreePath: {ERROR}", "ERROR", e);
626         return;
627     }
628 
629     return;
630 }
631 
632 void ItemUpdater::createActiveAssociation(const std::string& path)
633 {
634     assocs.emplace_back(
635         std::make_tuple(ACTIVE_FWD_ASSOCIATION, ACTIVE_REV_ASSOCIATION, path));
636     associations(assocs);
637 }
638 
639 void ItemUpdater::createFunctionalAssociation(const std::string& path)
640 {
641     assocs.emplace_back(std::make_tuple(FUNCTIONAL_FWD_ASSOCIATION,
642                                         FUNCTIONAL_REV_ASSOCIATION, path));
643     associations(assocs);
644 }
645 
646 void ItemUpdater::createUpdateableAssociation(const std::string& path)
647 {
648     assocs.emplace_back(std::make_tuple(UPDATEABLE_FWD_ASSOCIATION,
649                                         UPDATEABLE_REV_ASSOCIATION, path));
650     associations(assocs);
651 }
652 
653 void ItemUpdater::removeAssociations(const std::string& path)
654 {
655     for (auto iter = assocs.begin(); iter != assocs.end();)
656     {
657         if ((std::get<2>(*iter)).compare(path) == 0)
658         {
659             iter = assocs.erase(iter);
660             associations(assocs);
661         }
662         else
663         {
664             ++iter;
665         }
666     }
667 }
668 
669 bool ItemUpdater::isLowestPriority(uint8_t value)
670 {
671     for (const auto& intf : activations)
672     {
673         if (intf.second->redundancyPriority)
674         {
675             if (intf.second->redundancyPriority.get()->priority() < value)
676             {
677                 return false;
678             }
679         }
680     }
681     return true;
682 }
683 
684 void ItemUpdater::updateUbootEnvVars(const std::string& versionId)
685 {
686     auto it = versions.find(versionId);
687     if (it == versions.end())
688     {
689         return;
690     }
691     auto flashId = it->second->path();
692     helper.updateUbootVersionId(flashId);
693 }
694 
695 void ItemUpdater::resetUbootEnvVars()
696 {
697     decltype(activations.begin()->second->redundancyPriority.get()->priority())
698         lowestPriority = std::numeric_limits<uint8_t>::max();
699     decltype(activations.begin()->second->versionId) lowestPriorityVersion;
700     for (const auto& intf : activations)
701     {
702         if (!intf.second->redundancyPriority.get())
703         {
704             // Skip this version if the redundancyPriority is not initialized.
705             continue;
706         }
707 
708         if (intf.second->redundancyPriority.get()->priority() <= lowestPriority)
709         {
710             lowestPriority = intf.second->redundancyPriority.get()->priority();
711             lowestPriorityVersion = intf.second->versionId;
712         }
713     }
714 
715     // Update the U-boot environment variable to point to the lowest priority
716     updateUbootEnvVars(lowestPriorityVersion);
717 }
718 
719 void ItemUpdater::freeSpace([[maybe_unused]] const Activation& caller)
720 {
721 #ifdef BMC_STATIC_DUAL_IMAGE
722     // For the golden image case, always remove the version on the primary side
723     std::string versionIDtoErase;
724     for (const auto& iter : activations)
725     {
726         if (iter.second->redundancyPriority &&
727             iter.second->redundancyPriority.get()->priority() == 0)
728         {
729             versionIDtoErase = iter.second->versionId;
730             break;
731         }
732     }
733     if (!versionIDtoErase.empty())
734     {
735         erase(versionIDtoErase);
736     }
737     else
738     {
739         warning("Failed to find version to erase");
740     }
741 #else
742     //  Versions with the highest priority in front
743     std::priority_queue<std::pair<int, std::string>,
744                         std::vector<std::pair<int, std::string>>,
745                         std::less<std::pair<int, std::string>>>
746         versionsPQ;
747 
748     std::size_t count = 0;
749     for (const auto& iter : activations)
750     {
751         if ((iter.second.get()->activation() ==
752              server::Activation::Activations::Active) ||
753             (iter.second.get()->activation() ==
754              server::Activation::Activations::Failed))
755         {
756             count++;
757             // Don't put the functional version on the queue since we can't
758             // remove the "running" BMC version.
759             // If ACTIVE_BMC_MAX_ALLOWED <= 1, there is only one active BMC,
760             // so remove functional version as well.
761             // Don't delete the the Activation object that called this function.
762             if ((versions.find(iter.second->versionId)
763                      ->second->isFunctional() &&
764                  ACTIVE_BMC_MAX_ALLOWED > 1) ||
765                 (iter.second->versionId == caller.versionId))
766             {
767                 continue;
768             }
769 
770             // Failed activations don't have priority, assign them a large value
771             // for sorting purposes.
772             auto priority = 999;
773             if (iter.second.get()->activation() ==
774                     server::Activation::Activations::Active &&
775                 iter.second->redundancyPriority)
776             {
777                 priority = iter.second->redundancyPriority.get()->priority();
778             }
779 
780             versionsPQ.push(std::make_pair(priority, iter.second->versionId));
781         }
782     }
783 
784     // If the number of BMC versions is over ACTIVE_BMC_MAX_ALLOWED -1,
785     // remove the highest priority one(s).
786     while ((count >= ACTIVE_BMC_MAX_ALLOWED) && (!versionsPQ.empty()))
787     {
788         erase(versionsPQ.top().second);
789         versionsPQ.pop();
790         count--;
791     }
792 #endif
793 }
794 
795 void ItemUpdater::mirrorUbootToAlt()
796 {
797     helper.mirrorAlt();
798 }
799 
800 bool ItemUpdater::checkImage(const std::string& filePath,
801                              const std::vector<std::string>& imageList)
802 {
803     bool valid = true;
804 
805     for (auto& bmcImage : imageList)
806     {
807         fs::path file(filePath);
808         file /= bmcImage;
809         std::ifstream efile(file.c_str());
810         if (efile.good() != 1)
811         {
812             valid = false;
813             break;
814         }
815     }
816 
817     return valid;
818 }
819 
820 #ifdef HOST_BIOS_UPGRADE
821 void ItemUpdater::createBIOSObject()
822 {
823     std::string path = BIOS_OBJPATH;
824     // Get version id from last item in the path
825     auto pos = path.rfind("/");
826     if (pos == std::string::npos)
827     {
828         error("No version id found in object path {PATH}", "PATH", path);
829         return;
830     }
831 
832     createActiveAssociation(path);
833     createFunctionalAssociation(path);
834 
835     auto versionId = path.substr(pos + 1);
836     auto version = "null";
837     AssociationList assocs = {};
838     biosActivation = std::make_unique<Activation>(
839         bus, path, *this, versionId, server::Activation::Activations::Active,
840         assocs);
841     auto dummyErase = [](std::string /*entryId*/) {
842         // Do nothing;
843     };
844     biosVersion = std::make_unique<VersionClass>(
845         bus, path, version, VersionPurpose::Host, "", "",
846         std::bind(dummyErase, std::placeholders::_1), "");
847     biosVersion->deleteObject =
848         std::make_unique<phosphor::software::manager::Delete>(bus, path,
849                                                               *biosVersion);
850 }
851 #endif
852 
853 void ItemUpdater::getRunningSlot()
854 {
855     // Check /run/media/slot to get the slot number
856     constexpr auto slotFile = "/run/media/slot";
857     std::fstream f(slotFile, std::ios_base::in);
858     f >> runningImageSlot;
859 }
860 
861 } // namespace updater
862 } // namespace software
863 } // namespace phosphor
864