xref: /openbmc/phosphor-fan-presence/control/json/actions/override_fan_target.cpp (revision dfddd648cb81b27492afead4e2346f5fcd1397cb)
159af8caaSMike Capps /**
259af8caaSMike Capps  * Copyright © 2022 IBM Corporation
359af8caaSMike Capps  *
459af8caaSMike Capps  * Licensed under the Apache License, Version 2.0 (the "License");
559af8caaSMike Capps  * you may not use this file except in compliance with the License.
659af8caaSMike Capps  * You may obtain a copy of the License at
759af8caaSMike Capps  *
859af8caaSMike Capps  *     http://www.apache.org/licenses/LICENSE-2.0
959af8caaSMike Capps  *
1059af8caaSMike Capps  * Unless required by applicable law or agreed to in writing, software
1159af8caaSMike Capps  * distributed under the License is distributed on an "AS IS" BASIS,
1259af8caaSMike Capps  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
1359af8caaSMike Capps  * See the License for the specific language governing permissions and
1459af8caaSMike Capps  * limitations under the License.
1559af8caaSMike Capps  */
1659af8caaSMike Capps #include "override_fan_target.hpp"
1759af8caaSMike Capps 
1859af8caaSMike Capps #include "../manager.hpp"
1959af8caaSMike Capps #include "../zone.hpp"
2059af8caaSMike Capps #include "action.hpp"
2159af8caaSMike Capps #include "group.hpp"
2259af8caaSMike Capps 
2359af8caaSMike Capps #include <nlohmann/json.hpp>
2459af8caaSMike Capps 
25*2e22b235SMatt Spinler #include <format>
26*2e22b235SMatt Spinler 
2759af8caaSMike Capps namespace phosphor::fan::control::json
2859af8caaSMike Capps {
2959af8caaSMike Capps 
3059af8caaSMike Capps using json = nlohmann::json;
3159af8caaSMike Capps 
OverrideFanTarget(const json & jsonObj,const std::vector<Group> & groups)3259af8caaSMike Capps OverrideFanTarget::OverrideFanTarget(const json& jsonObj,
3359af8caaSMike Capps                                      const std::vector<Group>& groups) :
3459af8caaSMike Capps     ActionBase(jsonObj, groups)
3559af8caaSMike Capps {
3659af8caaSMike Capps     setCount(jsonObj);
3759af8caaSMike Capps     setState(jsonObj);
3859af8caaSMike Capps     setTarget(jsonObj);
3959af8caaSMike Capps     setFans(jsonObj);
4059af8caaSMike Capps }
4159af8caaSMike Capps 
run(Zone & zone)4259af8caaSMike Capps void OverrideFanTarget::run(Zone& zone)
4359af8caaSMike Capps {
4459af8caaSMike Capps     size_t numAtState = 0;
4559af8caaSMike Capps 
4659af8caaSMike Capps     for (const auto& group : _groups)
4759af8caaSMike Capps     {
4859af8caaSMike Capps         for (const auto& member : group.getMembers())
4959af8caaSMike Capps         {
5059af8caaSMike Capps             try
5159af8caaSMike Capps             {
5259af8caaSMike Capps                 if (Manager::getObjValueVariant(member, group.getInterface(),
5359af8caaSMike Capps                                                 group.getProperty()) == _state)
5459af8caaSMike Capps                 {
5559af8caaSMike Capps                     numAtState++;
5659af8caaSMike Capps 
5759af8caaSMike Capps                     if (numAtState >= _count)
5859af8caaSMike Capps                     {
5959af8caaSMike Capps                         break;
6059af8caaSMike Capps                     }
6159af8caaSMike Capps                 }
6259af8caaSMike Capps             }
6359af8caaSMike Capps             catch (const std::out_of_range&)
6459af8caaSMike Capps             {}
6559af8caaSMike Capps         }
6659af8caaSMike Capps 
6759af8caaSMike Capps         // lock the fans
6859af8caaSMike Capps         if (numAtState >= _count)
6959af8caaSMike Capps         {
7059af8caaSMike Capps             lockFans(zone);
7159af8caaSMike Capps             break;
7259af8caaSMike Capps         }
7359af8caaSMike Capps     }
7459af8caaSMike Capps 
7559af8caaSMike Capps     if (_locked && numAtState < _count)
7659af8caaSMike Capps     {
7759af8caaSMike Capps         unlockFans(zone);
7859af8caaSMike Capps     }
7959af8caaSMike Capps }
8059af8caaSMike Capps 
lockFans(Zone & zone)8159af8caaSMike Capps void OverrideFanTarget::lockFans(Zone& zone)
8259af8caaSMike Capps {
8359af8caaSMike Capps     if (!_locked)
8459af8caaSMike Capps     {
85*2e22b235SMatt Spinler         std::string fanList;
86*2e22b235SMatt Spinler         if (!_fans.empty())
87*2e22b235SMatt Spinler         {
88*2e22b235SMatt Spinler             fanList = std::accumulate(std::next(_fans.begin()), _fans.end(),
89*2e22b235SMatt Spinler                                       _fans[0], [](auto list, auto fan) {
90*2e22b235SMatt Spinler                                           return std::move(list) + ", " + fan;
91*2e22b235SMatt Spinler                                       });
92*2e22b235SMatt Spinler         }
93*2e22b235SMatt Spinler 
94*2e22b235SMatt Spinler         record(std::format("Adding fan target lock of {} on fans [{}] zone {}",
95*2e22b235SMatt Spinler                            _target, fanList, zone.getName()));
9659af8caaSMike Capps 
9759af8caaSMike Capps         for (auto& fan : _fans)
9859af8caaSMike Capps         {
9959af8caaSMike Capps             zone.lockFanTarget(fan, _target);
10059af8caaSMike Capps         }
10159af8caaSMike Capps 
10259af8caaSMike Capps         _locked = true;
10359af8caaSMike Capps     }
10459af8caaSMike Capps }
10559af8caaSMike Capps 
unlockFans(Zone & zone)10659af8caaSMike Capps void OverrideFanTarget::unlockFans(Zone& zone)
10759af8caaSMike Capps {
108*2e22b235SMatt Spinler     std::string fanList;
109*2e22b235SMatt Spinler     if (!_fans.empty())
110*2e22b235SMatt Spinler     {
111*2e22b235SMatt Spinler         fanList = std::accumulate(
112*2e22b235SMatt Spinler             std::next(_fans.begin()), _fans.end(), _fans[0],
113*2e22b235SMatt Spinler             [](auto list, auto fan) { return std::move(list) + ", " + fan; });
114*2e22b235SMatt Spinler     }
115*2e22b235SMatt Spinler 
116*2e22b235SMatt Spinler     record(std::format("Un-locking fan target {} on fans [{}] zone {}", _target,
117*2e22b235SMatt Spinler                        fanList, zone.getName()));
11859af8caaSMike Capps 
11959af8caaSMike Capps     // unlock all fans in this instance
12059af8caaSMike Capps     for (auto& fan : _fans)
12159af8caaSMike Capps     {
12259af8caaSMike Capps         zone.unlockFanTarget(fan, _target);
12359af8caaSMike Capps     }
12459af8caaSMike Capps 
12559af8caaSMike Capps     _locked = false;
12659af8caaSMike Capps }
12759af8caaSMike Capps 
setCount(const json & jsonObj)12859af8caaSMike Capps void OverrideFanTarget::setCount(const json& jsonObj)
12959af8caaSMike Capps {
13059af8caaSMike Capps     if (!jsonObj.contains("count"))
13159af8caaSMike Capps     {
13259af8caaSMike Capps         throw ActionParseError{ActionBase::getName(),
13359af8caaSMike Capps                                "Missing required count value"};
13459af8caaSMike Capps     }
13559af8caaSMike Capps     _count = jsonObj["count"].get<size_t>();
13659af8caaSMike Capps }
13759af8caaSMike Capps 
setState(const json & jsonObj)13859af8caaSMike Capps void OverrideFanTarget::setState(const json& jsonObj)
13959af8caaSMike Capps {
14059af8caaSMike Capps     if (!jsonObj.contains("state"))
14159af8caaSMike Capps     {
14259af8caaSMike Capps         throw ActionParseError{ActionBase::getName(),
14359af8caaSMike Capps                                "Missing required state value"};
14459af8caaSMike Capps     }
14559af8caaSMike Capps     _state = getJsonValue(jsonObj["state"]);
14659af8caaSMike Capps }
14759af8caaSMike Capps 
setTarget(const json & jsonObj)14859af8caaSMike Capps void OverrideFanTarget::setTarget(const json& jsonObj)
14959af8caaSMike Capps {
15059af8caaSMike Capps     if (!jsonObj.contains("target"))
15159af8caaSMike Capps     {
15259af8caaSMike Capps         throw ActionParseError{ActionBase::getName(),
15359af8caaSMike Capps                                "Missing required target value"};
15459af8caaSMike Capps     }
15559af8caaSMike Capps     _target = jsonObj["target"].get<uint64_t>();
15659af8caaSMike Capps }
15759af8caaSMike Capps 
setFans(const json & jsonObj)15859af8caaSMike Capps void OverrideFanTarget::setFans(const json& jsonObj)
15959af8caaSMike Capps {
16059af8caaSMike Capps     if (!jsonObj.contains("fans"))
16159af8caaSMike Capps     {
16259af8caaSMike Capps         throw ActionParseError{ActionBase::getName(),
16359af8caaSMike Capps                                "Missing required fans value"};
16459af8caaSMike Capps     }
16559af8caaSMike Capps 
16659af8caaSMike Capps     _fans = jsonObj["fans"].get<std::vector<std::string>>();
16759af8caaSMike Capps }
16859af8caaSMike Capps 
16959af8caaSMike Capps } // namespace phosphor::fan::control::json
170