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