17136a5aeSJames Feist /*
27136a5aeSJames Feist // Copyright (c) 2018 Intel Corporation
37136a5aeSJames Feist //
47136a5aeSJames Feist // Licensed under the Apache License, Version 2.0 (the "License");
57136a5aeSJames Feist // you may not use this file except in compliance with the License.
67136a5aeSJames Feist // You may obtain a copy of the License at
77136a5aeSJames Feist //
87136a5aeSJames Feist // http://www.apache.org/licenses/LICENSE-2.0
97136a5aeSJames Feist //
107136a5aeSJames Feist // Unless required by applicable law or agreed to in writing, software
117136a5aeSJames Feist // distributed under the License is distributed on an "AS IS" BASIS,
127136a5aeSJames Feist // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
137136a5aeSJames Feist // See the License for the specific language governing permissions and
147136a5aeSJames Feist // limitations under the License.
157136a5aeSJames Feist */
16*a552fe2fSChaul Ly #include "config.h"
17*a552fe2fSChaul Ly
187e952d9fSPatrick Venture #include "dbusconfiguration.hpp"
197136a5aeSJames Feist
200771659eSPatrick Venture #include "conf.hpp"
21ef1f8864SPatrick Venture #include "dbushelper.hpp"
22ef1f8864SPatrick Venture #include "dbusutil.hpp"
230c8223b5SJames Feist #include "util.hpp"
240771659eSPatrick Venture
251fe08952SJames Feist #include <boost/asio/steady_timer.hpp>
26a83a3eccSPatrick Venture #include <sdbusplus/bus.hpp>
27a83a3eccSPatrick Venture #include <sdbusplus/bus/match.hpp>
28a83a3eccSPatrick Venture #include <sdbusplus/exception.hpp>
29a83a3eccSPatrick Venture
30a83a3eccSPatrick Venture #include <algorithm>
3164f072a7SJames Feist #include <chrono>
3264f072a7SJames Feist #include <functional>
337136a5aeSJames Feist #include <iostream>
341fe08952SJames Feist #include <list>
357136a5aeSJames Feist #include <set>
367136a5aeSJames Feist #include <unordered_map>
371f802f5eSJames Feist #include <variant>
387136a5aeSJames Feist
39a076487aSPatrick Venture namespace pid_control
40a076487aSPatrick Venture {
41a076487aSPatrick Venture
427136a5aeSJames Feist constexpr const char* pidConfigurationInterface =
437136a5aeSJames Feist "xyz.openbmc_project.Configuration.Pid";
447136a5aeSJames Feist constexpr const char* objectManagerInterface =
457136a5aeSJames Feist "org.freedesktop.DBus.ObjectManager";
467136a5aeSJames Feist constexpr const char* pidZoneConfigurationInterface =
477136a5aeSJames Feist "xyz.openbmc_project.Configuration.Pid.Zone";
4822c257abSJames Feist constexpr const char* stepwiseConfigurationInterface =
4922c257abSJames Feist "xyz.openbmc_project.Configuration.Stepwise";
50f0096a0cSJames Feist constexpr const char* thermalControlIface =
51f0096a0cSJames Feist "xyz.openbmc_project.Control.ThermalMode";
527136a5aeSJames Feist constexpr const char* sensorInterface = "xyz.openbmc_project.Sensor.Value";
530911bfe7SPatrick Venture constexpr const char* defaultPwmInterface =
540911bfe7SPatrick Venture "xyz.openbmc_project.Control.FanPwm";
557136a5aeSJames Feist
56991ebd80SJames Feist using Association = std::tuple<std::string, std::string, std::string>;
57991ebd80SJames Feist using Associations = std::vector<Association>;
58991ebd80SJames Feist
595ec20270SJames Feist namespace thresholds
605ec20270SJames Feist {
615ec20270SJames Feist constexpr const char* warningInterface =
625ec20270SJames Feist "xyz.openbmc_project.Sensor.Threshold.Warning";
635ec20270SJames Feist constexpr const char* criticalInterface =
645ec20270SJames Feist "xyz.openbmc_project.Sensor.Threshold.Critical";
655ec20270SJames Feist const std::array<const char*, 4> types = {"CriticalLow", "CriticalHigh",
665ec20270SJames Feist "WarningLow", "WarningHigh"};
675ec20270SJames Feist
685ec20270SJames Feist } // namespace thresholds
695ec20270SJames Feist
707136a5aeSJames Feist namespace dbus_configuration
717136a5aeSJames Feist {
getSensorNameFromPath(const std::string & dbusPath)72f3b04fd3SJason Ling using SensorInterfaceType = std::pair<std::string, std::string>;
73f3b04fd3SJason Ling
74f3b04fd3SJason Ling inline std::string getSensorNameFromPath(const std::string& dbusPath)
75f3b04fd3SJason Ling {
76f3b04fd3SJason Ling return dbusPath.substr(dbusPath.find_last_of("/") + 1);
sensorNameToDbusName(const std::string & sensorName)77f3b04fd3SJason Ling }
78f3b04fd3SJason Ling
79f3b04fd3SJason Ling inline std::string sensorNameToDbusName(const std::string& sensorName)
80f3b04fd3SJason Ling {
81f3b04fd3SJason Ling std::string retString = sensorName;
82f3b04fd3SJason Ling std::replace(retString.begin(), retString.end(), ' ', '_');
83f3b04fd3SJason Ling return retString;
84f3b04fd3SJason Ling }
855ec20270SJames Feist
86b228bc30SPatrick Williams std::vector<std::string> getSelectedProfiles(sdbusplus::bus_t& bus)
87f0096a0cSJames Feist {
88f0096a0cSJames Feist std::vector<std::string> ret;
89bd63bcacSPatrick Williams auto mapper =
90bd63bcacSPatrick Williams bus.new_method_call("xyz.openbmc_project.ObjectMapper",
91f0096a0cSJames Feist "/xyz/openbmc_project/object_mapper",
92bd63bcacSPatrick Williams "xyz.openbmc_project.ObjectMapper", "GetSubTree");
93f0096a0cSJames Feist mapper.append("/", 0, std::array<const char*, 1>{thermalControlIface});
94f0096a0cSJames Feist std::unordered_map<
95f0096a0cSJames Feist std::string, std::unordered_map<std::string, std::vector<std::string>>>
96f0096a0cSJames Feist respData;
97f0096a0cSJames Feist
98f0096a0cSJames Feist try
99f0096a0cSJames Feist {
100f0096a0cSJames Feist auto resp = bus.call(mapper);
101f0096a0cSJames Feist resp.read(respData);
102f0096a0cSJames Feist }
1030001ee02SPatrick Williams catch (const sdbusplus::exception_t&)
104f0096a0cSJames Feist {
105f0096a0cSJames Feist // can't do anything without mapper call data
106f0096a0cSJames Feist throw std::runtime_error("ObjectMapper Call Failure");
107f0096a0cSJames Feist }
108f0096a0cSJames Feist if (respData.empty())
109f0096a0cSJames Feist {
110f0096a0cSJames Feist // if the user has profiles but doesn't expose the interface to select
111f0096a0cSJames Feist // one, just go ahead without using profiles
112f0096a0cSJames Feist return ret;
113f0096a0cSJames Feist }
114f0096a0cSJames Feist
115f0096a0cSJames Feist // assumption is that we should only have a small handful of selected
116f0096a0cSJames Feist // profiles at a time (probably only 1), so calling each individually should
117f0096a0cSJames Feist // not incur a large cost
118f0096a0cSJames Feist for (const auto& objectPair : respData)
119f0096a0cSJames Feist {
120f0096a0cSJames Feist const std::string& path = objectPair.first;
121f0096a0cSJames Feist for (const auto& ownerPair : objectPair.second)
122f0096a0cSJames Feist {
123f0096a0cSJames Feist const std::string& busName = ownerPair.first;
124f0096a0cSJames Feist auto getProfile =
125f0096a0cSJames Feist bus.new_method_call(busName.c_str(), path.c_str(),
126f0096a0cSJames Feist "org.freedesktop.DBus.Properties", "Get");
127f0096a0cSJames Feist getProfile.append(thermalControlIface, "Current");
128f0096a0cSJames Feist std::variant<std::string> variantResp;
129f0096a0cSJames Feist try
130f0096a0cSJames Feist {
131f0096a0cSJames Feist auto resp = bus.call(getProfile);
132f0096a0cSJames Feist resp.read(variantResp);
133f0096a0cSJames Feist }
1340001ee02SPatrick Williams catch (const sdbusplus::exception_t&)
135f0096a0cSJames Feist {
136f0096a0cSJames Feist throw std::runtime_error("Failure getting profile");
137f0096a0cSJames Feist }
138f0096a0cSJames Feist std::string mode = std::get<std::string>(variantResp);
139f0096a0cSJames Feist ret.emplace_back(std::move(mode));
140f0096a0cSJames Feist }
141f0096a0cSJames Feist }
14239199b4dSPatrick Venture if constexpr (pid_control::conf::DEBUG)
143f0096a0cSJames Feist {
144f0096a0cSJames Feist std::cout << "Profiles selected: ";
145f0096a0cSJames Feist for (const auto& profile : ret)
146f0096a0cSJames Feist {
147f0096a0cSJames Feist std::cout << profile << " ";
148f0096a0cSJames Feist }
149f0096a0cSJames Feist std::cout << "\n";
150f0096a0cSJames Feist }
151f0096a0cSJames Feist return ret;
eventHandler(sd_bus_message * m,void * context,sd_bus_error *)152f0096a0cSJames Feist }
153f0096a0cSJames Feist
154991ebd80SJames Feist int eventHandler(sd_bus_message* m, void* context, sd_bus_error*)
1557136a5aeSJames Feist {
156991ebd80SJames Feist if (context == nullptr || m == nullptr)
1571fe08952SJames Feist {
1581fe08952SJames Feist throw std::runtime_error("Invalid match");
1591fe08952SJames Feist }
160991ebd80SJames Feist
161991ebd80SJames Feist // we skip associations because the mapper populates these, not the sensors
16210e46efaSJosh Lehan const std::array<const char*, 2> skipList = {
16310e46efaSJosh Lehan "xyz.openbmc_project.Association",
16410e46efaSJosh Lehan "xyz.openbmc_project.Association.Definitions"};
165991ebd80SJames Feist
166b228bc30SPatrick Williams sdbusplus::message_t message(m);
167991ebd80SJames Feist if (std::string(message.get_member()) == "InterfacesAdded")
168991ebd80SJames Feist {
169991ebd80SJames Feist sdbusplus::message::object_path path;
170991ebd80SJames Feist std::unordered_map<
171991ebd80SJames Feist std::string,
172991ebd80SJames Feist std::unordered_map<std::string, std::variant<Associations, bool>>>
173991ebd80SJames Feist data;
174991ebd80SJames Feist
175991ebd80SJames Feist message.read(path, data);
176991ebd80SJames Feist
177991ebd80SJames Feist for (const char* skip : skipList)
178991ebd80SJames Feist {
179991ebd80SJames Feist auto find = data.find(skip);
180991ebd80SJames Feist if (find != data.end())
181991ebd80SJames Feist {
182991ebd80SJames Feist data.erase(find);
183991ebd80SJames Feist if (data.empty())
184991ebd80SJames Feist {
185991ebd80SJames Feist return 1;
186991ebd80SJames Feist }
187991ebd80SJames Feist }
188991ebd80SJames Feist }
18910e46efaSJosh Lehan
19010e46efaSJosh Lehan if constexpr (pid_control::conf::DEBUG)
19110e46efaSJosh Lehan {
19210e46efaSJosh Lehan std::cout << "New config detected: " << path.str << std::endl;
19310e46efaSJosh Lehan for (auto& d : data)
19410e46efaSJosh Lehan {
19510e46efaSJosh Lehan std::cout << "\tdata is " << d.first << std::endl;
19610e46efaSJosh Lehan for (auto& second : d.second)
19710e46efaSJosh Lehan {
19810e46efaSJosh Lehan std::cout << "\t\tdata is " << second.first << std::endl;
19910e46efaSJosh Lehan }
20010e46efaSJosh Lehan }
20110e46efaSJosh Lehan }
202991ebd80SJames Feist }
203991ebd80SJames Feist
2041fe08952SJames Feist boost::asio::steady_timer* timer =
2051fe08952SJames Feist static_cast<boost::asio::steady_timer*>(context);
2061fe08952SJames Feist
2071fe08952SJames Feist // do a brief sleep as we tend to get a bunch of these events at
2081fe08952SJames Feist // once
2091fe08952SJames Feist timer->expires_after(std::chrono::seconds(2));
2101fe08952SJames Feist timer->async_wait([](const boost::system::error_code ec) {
2111fe08952SJames Feist if (ec == boost::asio::error::operation_aborted)
2121fe08952SJames Feist {
2131fe08952SJames Feist /* another timer started*/
2141fe08952SJames Feist return;
2151fe08952SJames Feist }
2161fe08952SJames Feist
2171fe08952SJames Feist std::cout << "New configuration detected, reloading\n.";
218298a95cbSYong Li tryRestartControlLoops();
2191fe08952SJames Feist });
2201fe08952SJames Feist
2211fe08952SJames Feist return 1;
2221fe08952SJames Feist }
2231fe08952SJames Feist
224b228bc30SPatrick Williams void createMatches(sdbusplus::bus_t& bus, boost::asio::steady_timer& timer)
2251fe08952SJames Feist {
2261fe08952SJames Feist // this is a list because the matches can't be moved
227b228bc30SPatrick Williams static std::list<sdbusplus::bus::match_t> matches;
2281fe08952SJames Feist
2293987c8b4SJames Feist const std::array<std::string, 4> interfaces = {
2303987c8b4SJames Feist thermalControlIface, pidConfigurationInterface,
2313987c8b4SJames Feist pidZoneConfigurationInterface, stepwiseConfigurationInterface};
2321fe08952SJames Feist
2331fe08952SJames Feist // this list only needs to be created once
2341fe08952SJames Feist if (!matches.empty())
2351fe08952SJames Feist {
2361fe08952SJames Feist return;
2371fe08952SJames Feist }
2381fe08952SJames Feist
2391fe08952SJames Feist // we restart when the configuration changes or there are new sensors
2401fe08952SJames Feist for (const auto& interface : interfaces)
2411fe08952SJames Feist {
2421fe08952SJames Feist matches.emplace_back(
2431fe08952SJames Feist bus,
2441fe08952SJames Feist "type='signal',member='PropertiesChanged',arg0namespace='" +
2451fe08952SJames Feist interface + "'",
2461fe08952SJames Feist eventHandler, &timer);
2471fe08952SJames Feist }
2481fe08952SJames Feist matches.emplace_back(
2491fe08952SJames Feist bus,
2501fe08952SJames Feist "type='signal',member='InterfacesAdded',arg0path='/xyz/openbmc_project/"
2511fe08952SJames Feist "sensors/'",
2521fe08952SJames Feist eventHandler, &timer);
253c2a311b0SJinliang Wang matches.emplace_back(bus,
254c2a311b0SJinliang Wang "type='signal',member='InterfacesRemoved',arg0path='/"
255c2a311b0SJinliang Wang "xyz/openbmc_project/sensors/'",
256c2a311b0SJinliang Wang eventHandler, &timer);
2571fe08952SJames Feist }
2581fe08952SJames Feist
2596fc301fbSJason Ling /**
2606fc301fbSJason Ling * retrieve an attribute from the pid configuration map
2616fc301fbSJason Ling * @param[in] base - the PID configuration map, keys are the attributes and
2626fc301fbSJason Ling * value is the variant associated with that attribute.
2636fc301fbSJason Ling * @param attributeName - the name of the attribute
2646fc301fbSJason Ling * @return a variant holding the value associated with a key
getPIDAttribute(const std::unordered_map<std::string,DbusVariantType> & base,const std::string & attributeName)2656fc301fbSJason Ling * @throw runtime_error : attributeName is not in base
2666fc301fbSJason Ling */
2676fc301fbSJason Ling inline DbusVariantType getPIDAttribute(
2686fc301fbSJason Ling const std::unordered_map<std::string, DbusVariantType>& base,
2696fc301fbSJason Ling const std::string& attributeName)
2706fc301fbSJason Ling {
2716fc301fbSJason Ling auto search = base.find(attributeName);
2726fc301fbSJason Ling if (search == base.end())
2736fc301fbSJason Ling {
2746fc301fbSJason Ling throw std::runtime_error("missing attribute " + attributeName);
2756fc301fbSJason Ling }
2766fc301fbSJason Ling return search->second;
2776fc301fbSJason Ling }
2786fc301fbSJason Ling
279239aa7d7SHarvey Wu inline void getCycleTimeSetting(
280239aa7d7SHarvey Wu const std::unordered_map<std::string, DbusVariantType>& zone,
281239aa7d7SHarvey Wu const int zoneIndex, const std::string& attributeName, uint64_t& value)
282239aa7d7SHarvey Wu {
283239aa7d7SHarvey Wu auto findAttributeName = zone.find(attributeName);
284239aa7d7SHarvey Wu if (findAttributeName != zone.end())
285239aa7d7SHarvey Wu {
286bd63bcacSPatrick Williams double tmpAttributeValue =
287bd63bcacSPatrick Williams std::visit(VariantToDoubleVisitor(), zone.at(attributeName));
288239aa7d7SHarvey Wu if (tmpAttributeValue >= 1.0)
289239aa7d7SHarvey Wu {
290239aa7d7SHarvey Wu value = static_cast<uint64_t>(tmpAttributeValue);
291239aa7d7SHarvey Wu }
292239aa7d7SHarvey Wu else
293239aa7d7SHarvey Wu {
294239aa7d7SHarvey Wu std::cerr << "Zone " << zoneIndex << ": " << attributeName
295239aa7d7SHarvey Wu << " is invalid. Use default " << value << " ms\n";
296239aa7d7SHarvey Wu }
297239aa7d7SHarvey Wu }
298239aa7d7SHarvey Wu else
299239aa7d7SHarvey Wu {
300239aa7d7SHarvey Wu std::cerr << "Zone " << zoneIndex << ": " << attributeName
301239aa7d7SHarvey Wu << " cannot find setting. Use default " << value << " ms\n";
302239aa7d7SHarvey Wu }
populatePidInfo(sdbusplus::bus_t & bus,const std::unordered_map<std::string,DbusVariantType> & base,conf::ControllerInfo & info,const std::string * thresholdProperty,const std::map<std::string,conf::SensorConfig> & sensorConfig)303239aa7d7SHarvey Wu }
304239aa7d7SHarvey Wu
3055ec20270SJames Feist void populatePidInfo(
306a1ae4fa1SHarvey.Wu [[maybe_unused]] sdbusplus::bus_t& bus,
3075ec20270SJames Feist const std::unordered_map<std::string, DbusVariantType>& base,
3081df9e879SPatrick Venture conf::ControllerInfo& info, const std::string* thresholdProperty,
3097382318fSPatrick Venture const std::map<std::string, conf::SensorConfig>& sensorConfig)
3105ec20270SJames Feist {
3116fc301fbSJason Ling info.type = std::get<std::string>(getPIDAttribute(base, "Class"));
3125ec20270SJames Feist if (info.type == "fan")
3135ec20270SJames Feist {
3145ec20270SJames Feist info.setpoint = 0;
3155ec20270SJames Feist }
3165ec20270SJames Feist else
3175ec20270SJames Feist {
3186fc301fbSJason Ling info.setpoint = std::visit(VariantToDoubleVisitor(),
3196fc301fbSJason Ling getPIDAttribute(base, "SetPoint"));
3205ec20270SJames Feist }
3215ec20270SJames Feist
3229fe3a3c7Sykchiu int failsafepercent = 0;
3239fe3a3c7Sykchiu auto findFailSafe = base.find("FailSafePercent");
3249fe3a3c7Sykchiu if (findFailSafe != base.end())
3259fe3a3c7Sykchiu {
3269fe3a3c7Sykchiu failsafepercent = std::visit(VariantToDoubleVisitor(),
3279fe3a3c7Sykchiu getPIDAttribute(base, "FailSafePercent"));
3289fe3a3c7Sykchiu }
3299fe3a3c7Sykchiu info.failSafePercent = failsafepercent;
3309fe3a3c7Sykchiu
3315ec20270SJames Feist if (thresholdProperty != nullptr)
3325ec20270SJames Feist {
3335ec20270SJames Feist std::string interface;
3345ec20270SJames Feist if (*thresholdProperty == "WarningHigh" ||
3355ec20270SJames Feist *thresholdProperty == "WarningLow")
3365ec20270SJames Feist {
3375ec20270SJames Feist interface = thresholds::warningInterface;
3385ec20270SJames Feist }
3395ec20270SJames Feist else
3405ec20270SJames Feist {
3415ec20270SJames Feist interface = thresholds::criticalInterface;
3425ec20270SJames Feist }
34331058fd3SJosh Lehan
34431058fd3SJosh Lehan // Although this checks only the first vector element for the
34531058fd3SJosh Lehan // named threshold, it is OK, because the SetPointOffset parser
34631058fd3SJosh Lehan // splits up the input into individual vectors, each with only a
34731058fd3SJosh Lehan // single element, if it detects that SetPointOffset is in use.
34831058fd3SJosh Lehan const std::string& path =
34931058fd3SJosh Lehan sensorConfig.at(info.inputs.front().name).readPath;
3505ec20270SJames Feist
3518729eb98SPatrick Venture DbusHelper helper(sdbusplus::bus::new_system());
3529b93692dSPatrick Venture std::string service = helper.getService(interface, path);
3535ec20270SJames Feist double reading = 0;
3545ec20270SJames Feist try
3555ec20270SJames Feist {
3569b93692dSPatrick Venture helper.getProperty(service, path, interface, *thresholdProperty,
3579b93692dSPatrick Venture reading);
3585ec20270SJames Feist }
359b228bc30SPatrick Williams catch (const sdbusplus::exception_t& ex)
3605ec20270SJames Feist {
3615ec20270SJames Feist // unsupported threshold, leaving reading at 0
3625ec20270SJames Feist }
3635ec20270SJames Feist
3645ec20270SJames Feist info.setpoint += reading;
3655ec20270SJames Feist }
3665ec20270SJames Feist
3675ec20270SJames Feist info.pidInfo.ts = 1.0; // currently unused
3686fc301fbSJason Ling info.pidInfo.proportionalCoeff = std::visit(
3696fc301fbSJason Ling VariantToDoubleVisitor(), getPIDAttribute(base, "PCoefficient"));
3706fc301fbSJason Ling info.pidInfo.integralCoeff = std::visit(
3716fc301fbSJason Ling VariantToDoubleVisitor(), getPIDAttribute(base, "ICoefficient"));
372c612c051SJosh Lehan // DCoefficient is below, it is optional, same reason as in buildjson.cpp
3736fc301fbSJason Ling info.pidInfo.feedFwdOffset = std::visit(
3746fc301fbSJason Ling VariantToDoubleVisitor(), getPIDAttribute(base, "FFOffCoefficient"));
3756fc301fbSJason Ling info.pidInfo.feedFwdGain = std::visit(
3766fc301fbSJason Ling VariantToDoubleVisitor(), getPIDAttribute(base, "FFGainCoefficient"));
3776fc301fbSJason Ling info.pidInfo.integralLimit.max = std::visit(
3786fc301fbSJason Ling VariantToDoubleVisitor(), getPIDAttribute(base, "ILimitMax"));
3796fc301fbSJason Ling info.pidInfo.integralLimit.min = std::visit(
3806fc301fbSJason Ling VariantToDoubleVisitor(), getPIDAttribute(base, "ILimitMin"));
3816fc301fbSJason Ling info.pidInfo.outLim.max = std::visit(VariantToDoubleVisitor(),
3826fc301fbSJason Ling getPIDAttribute(base, "OutLimitMax"));
3836fc301fbSJason Ling info.pidInfo.outLim.min = std::visit(VariantToDoubleVisitor(),
3846fc301fbSJason Ling getPIDAttribute(base, "OutLimitMin"));
385bd63bcacSPatrick Williams info.pidInfo.slewNeg =
386bd63bcacSPatrick Williams std::visit(VariantToDoubleVisitor(), getPIDAttribute(base, "SlewNeg"));
387bd63bcacSPatrick Williams info.pidInfo.slewPos =
388bd63bcacSPatrick Williams std::visit(VariantToDoubleVisitor(), getPIDAttribute(base, "SlewPos"));
389c612c051SJosh Lehan
3909788963cSDelphine CC Chiu bool checkHysterWithSetpt = false;
3915ec20270SJames Feist double negativeHysteresis = 0;
3925ec20270SJames Feist double positiveHysteresis = 0;
393c612c051SJosh Lehan double derivativeCoeff = 0;
3945ec20270SJames Feist
3955d897e2aSDelphine CC Chiu auto findCheckHysterFlag = base.find("CheckHysteresisWithSetpoint");
3965ec20270SJames Feist auto findNeg = base.find("NegativeHysteresis");
3975ec20270SJames Feist auto findPos = base.find("PositiveHysteresis");
398c612c051SJosh Lehan auto findDerivative = base.find("DCoefficient");
3995ec20270SJames Feist
4009788963cSDelphine CC Chiu if (findCheckHysterFlag != base.end())
4019788963cSDelphine CC Chiu {
4029788963cSDelphine CC Chiu checkHysterWithSetpt = std::get<bool>(findCheckHysterFlag->second);
4039788963cSDelphine CC Chiu }
4045ec20270SJames Feist if (findNeg != base.end())
4055ec20270SJames Feist {
406bd63bcacSPatrick Williams negativeHysteresis =
407bd63bcacSPatrick Williams std::visit(VariantToDoubleVisitor(), findNeg->second);
4085ec20270SJames Feist }
4095ec20270SJames Feist if (findPos != base.end())
4105ec20270SJames Feist {
411bd63bcacSPatrick Williams positiveHysteresis =
412bd63bcacSPatrick Williams std::visit(VariantToDoubleVisitor(), findPos->second);
4135ec20270SJames Feist }
414c612c051SJosh Lehan if (findDerivative != base.end())
415c612c051SJosh Lehan {
416bd63bcacSPatrick Williams derivativeCoeff =
417bd63bcacSPatrick Williams std::visit(VariantToDoubleVisitor(), findDerivative->second);
418c612c051SJosh Lehan }
419c612c051SJosh Lehan
4209788963cSDelphine CC Chiu info.pidInfo.checkHysterWithSetpt = checkHysterWithSetpt;
4215ec20270SJames Feist info.pidInfo.negativeHysteresis = negativeHysteresis;
4225ec20270SJames Feist info.pidInfo.positiveHysteresis = positiveHysteresis;
423c612c051SJosh Lehan info.pidInfo.derivativeCoeff = derivativeCoeff;
4245ec20270SJames Feist }
4255ec20270SJames Feist
426b228bc30SPatrick Williams bool init(sdbusplus::bus_t& bus, boost::asio::steady_timer& timer,
4277382318fSPatrick Venture std::map<std::string, conf::SensorConfig>& sensorConfig,
4287382318fSPatrick Venture std::map<int64_t, conf::PIDConf>& zoneConfig,
4297382318fSPatrick Venture std::map<int64_t, conf::ZoneConfig>& zoneDetailsConfig)
4301fe08952SJames Feist {
4311fe08952SJames Feist sensorConfig.clear();
4321fe08952SJames Feist zoneConfig.clear();
4331fe08952SJames Feist zoneDetailsConfig.clear();
4341fe08952SJames Feist
4351fe08952SJames Feist createMatches(bus, timer);
4361fe08952SJames Feist
437bd63bcacSPatrick Williams auto mapper =
438bd63bcacSPatrick Williams bus.new_method_call("xyz.openbmc_project.ObjectMapper",
4397136a5aeSJames Feist "/xyz/openbmc_project/object_mapper",
440bd63bcacSPatrick Williams "xyz.openbmc_project.ObjectMapper", "GetSubTree");
441bd63bcacSPatrick Williams mapper.append(
442bd63bcacSPatrick Williams "/", 0,
4430911bfe7SPatrick Venture std::array<const char*, 6>{
4440911bfe7SPatrick Venture objectManagerInterface, pidConfigurationInterface,
445bd63bcacSPatrick Williams pidZoneConfigurationInterface, stepwiseConfigurationInterface,
446bd63bcacSPatrick Williams sensorInterface, defaultPwmInterface});
44722c257abSJames Feist std::unordered_map<
44822c257abSJames Feist std::string, std::unordered_map<std::string, std::vector<std::string>>>
44922c257abSJames Feist respData;
45022c257abSJames Feist try
45122c257abSJames Feist {
4527136a5aeSJames Feist auto resp = bus.call(mapper);
4537136a5aeSJames Feist resp.read(respData);
45422c257abSJames Feist }
4550001ee02SPatrick Williams catch (const sdbusplus::exception_t&)
45622c257abSJames Feist {
45722c257abSJames Feist // can't do anything without mapper call data
45822c257abSJames Feist throw std::runtime_error("ObjectMapper Call Failure");
45922c257abSJames Feist }
46022c257abSJames Feist
4617136a5aeSJames Feist if (respData.empty())
4627136a5aeSJames Feist {
46322c257abSJames Feist // can't do anything without mapper call data
4647136a5aeSJames Feist throw std::runtime_error("No configuration data available from Mapper");
4657136a5aeSJames Feist }
4667136a5aeSJames Feist // create a map of pair of <has pid configuration, ObjectManager path>
4677136a5aeSJames Feist std::unordered_map<std::string, std::pair<bool, std::string>> owners;
4687136a5aeSJames Feist // and a map of <path, interface> for sensors
4697136a5aeSJames Feist std::unordered_map<std::string, std::string> sensors;
4707136a5aeSJames Feist for (const auto& objectPair : respData)
4717136a5aeSJames Feist {
4727136a5aeSJames Feist for (const auto& ownerPair : objectPair.second)
4737136a5aeSJames Feist {
4747136a5aeSJames Feist auto& owner = owners[ownerPair.first];
4757136a5aeSJames Feist for (const std::string& interface : ownerPair.second)
4767136a5aeSJames Feist {
4777136a5aeSJames Feist if (interface == objectManagerInterface)
4787136a5aeSJames Feist {
4797136a5aeSJames Feist owner.second = objectPair.first;
4807136a5aeSJames Feist }
4817136a5aeSJames Feist if (interface == pidConfigurationInterface ||
48222c257abSJames Feist interface == pidZoneConfigurationInterface ||
48322c257abSJames Feist interface == stepwiseConfigurationInterface)
4847136a5aeSJames Feist {
4857136a5aeSJames Feist owner.first = true;
4867136a5aeSJames Feist }
4870911bfe7SPatrick Venture if (interface == sensorInterface ||
4880911bfe7SPatrick Venture interface == defaultPwmInterface)
4897136a5aeSJames Feist {
4907136a5aeSJames Feist // we're not interested in pwm sensors, just pwm control
4917136a5aeSJames Feist if (interface == sensorInterface &&
4927136a5aeSJames Feist objectPair.first.find("pwm") != std::string::npos)
4937136a5aeSJames Feist {
4947136a5aeSJames Feist continue;
4957136a5aeSJames Feist }
4967136a5aeSJames Feist sensors[objectPair.first] = interface;
4977136a5aeSJames Feist }
4987136a5aeSJames Feist }
4997136a5aeSJames Feist }
5007136a5aeSJames Feist }
5017136a5aeSJames Feist ManagedObjectType configurations;
5027136a5aeSJames Feist for (const auto& owner : owners)
5037136a5aeSJames Feist {
5047136a5aeSJames Feist // skip if no pid configuration (means probably a sensor)
5057136a5aeSJames Feist if (!owner.second.first)
5067136a5aeSJames Feist {
5077136a5aeSJames Feist continue;
5087136a5aeSJames Feist }
5097136a5aeSJames Feist auto endpoint = bus.new_method_call(
5107136a5aeSJames Feist owner.first.c_str(), owner.second.second.c_str(),
5117136a5aeSJames Feist "org.freedesktop.DBus.ObjectManager", "GetManagedObjects");
51222c257abSJames Feist ManagedObjectType configuration;
51322c257abSJames Feist try
51422c257abSJames Feist {
5157ca8887cSManojkiran Eda auto response = bus.call(endpoint);
5167ca8887cSManojkiran Eda response.read(configuration);
51722c257abSJames Feist }
5180001ee02SPatrick Williams catch (const sdbusplus::exception_t&)
51922c257abSJames Feist {
52022c257abSJames Feist // this shouldn't happen, probably means daemon crashed
521bd63bcacSPatrick Williams throw std::runtime_error(
522bd63bcacSPatrick Williams "Error getting managed objects from " + owner.first);
52322c257abSJames Feist }
52422c257abSJames Feist
5257136a5aeSJames Feist for (auto& pathPair : configuration)
5267136a5aeSJames Feist {
5277136a5aeSJames Feist if (pathPair.second.find(pidConfigurationInterface) !=
5287136a5aeSJames Feist pathPair.second.end() ||
5297136a5aeSJames Feist pathPair.second.find(pidZoneConfigurationInterface) !=
53022c257abSJames Feist pathPair.second.end() ||
53122c257abSJames Feist pathPair.second.find(stepwiseConfigurationInterface) !=
5327136a5aeSJames Feist pathPair.second.end())
5337136a5aeSJames Feist {
5347136a5aeSJames Feist configurations.emplace(pathPair);
5357136a5aeSJames Feist }
536f0096a0cSJames Feist }
537f0096a0cSJames Feist }
538f0096a0cSJames Feist
539f0096a0cSJames Feist // remove controllers from config that aren't in the current profile(s)
540f0096a0cSJames Feist std::vector<std::string> selectedProfiles = getSelectedProfiles(bus);
541f0096a0cSJames Feist if (selectedProfiles.size())
542f0096a0cSJames Feist {
5433987c8b4SJames Feist for (auto pathIt = configurations.begin();
5443987c8b4SJames Feist pathIt != configurations.end();)
545f0096a0cSJames Feist {
5463987c8b4SJames Feist for (auto confIt = pathIt->second.begin();
5473987c8b4SJames Feist confIt != pathIt->second.end();)
5483987c8b4SJames Feist {
5493987c8b4SJames Feist auto profilesFind = confIt->second.find("Profiles");
5503987c8b4SJames Feist if (profilesFind == confIt->second.end())
5513987c8b4SJames Feist {
5523987c8b4SJames Feist confIt++;
5533987c8b4SJames Feist continue; // if no profiles selected, apply always
5543987c8b4SJames Feist }
5553987c8b4SJames Feist auto profiles =
5563987c8b4SJames Feist std::get<std::vector<std::string>>(profilesFind->second);
5573987c8b4SJames Feist if (profiles.empty())
5583987c8b4SJames Feist {
5593987c8b4SJames Feist confIt++;
5603987c8b4SJames Feist continue;
561f0096a0cSJames Feist }
562f0096a0cSJames Feist
5633987c8b4SJames Feist bool found = false;
5643987c8b4SJames Feist for (const std::string& profile : profiles)
565f0096a0cSJames Feist {
5663987c8b4SJames Feist if (std::find(selectedProfiles.begin(),
567bd63bcacSPatrick Williams selectedProfiles.end(), profile) !=
568bd63bcacSPatrick Williams selectedProfiles.end())
569f0096a0cSJames Feist {
5703987c8b4SJames Feist found = true;
5713987c8b4SJames Feist break;
5723987c8b4SJames Feist }
5733987c8b4SJames Feist }
5743987c8b4SJames Feist if (found)
5753987c8b4SJames Feist {
5763987c8b4SJames Feist confIt++;
577f0096a0cSJames Feist }
578f0096a0cSJames Feist else
579f0096a0cSJames Feist {
5803987c8b4SJames Feist confIt = pathIt->second.erase(confIt);
581f0096a0cSJames Feist }
582f0096a0cSJames Feist }
5833987c8b4SJames Feist if (pathIt->second.empty())
584f0096a0cSJames Feist {
5853987c8b4SJames Feist pathIt = configurations.erase(pathIt);
586f0096a0cSJames Feist }
587f0096a0cSJames Feist else
588f0096a0cSJames Feist {
5893987c8b4SJames Feist pathIt++;
590f0096a0cSJames Feist }
5917136a5aeSJames Feist }
5927136a5aeSJames Feist }
5938c3c51eeSJames Feist
594998fbe67SJosh Lehan // On D-Bus, although not necessary,
595998fbe67SJosh Lehan // having the "zoneID" field can still be useful,
596998fbe67SJosh Lehan // as it is used for diagnostic messages,
597998fbe67SJosh Lehan // logging file names, and so on.
598998fbe67SJosh Lehan // Accept optional "ZoneIndex" parameter to explicitly specify.
599998fbe67SJosh Lehan // If not present, or not unique, auto-assign index,
600998fbe67SJosh Lehan // using 0-based numbering, ensuring uniqueness.
601998fbe67SJosh Lehan std::map<std::string, int64_t> foundZones;
6027136a5aeSJames Feist for (const auto& configuration : configurations)
6037136a5aeSJames Feist {
6047136a5aeSJames Feist auto findZone =
6057136a5aeSJames Feist configuration.second.find(pidZoneConfigurationInterface);
6067136a5aeSJames Feist if (findZone != configuration.second.end())
6077136a5aeSJames Feist {
6087136a5aeSJames Feist const auto& zone = findZone->second;
609ffd418bbSJames Feist
6101f802f5eSJames Feist const std::string& name = std::get<std::string>(zone.at("Name"));
611998fbe67SJosh Lehan
612998fbe67SJosh Lehan auto findZoneIndex = zone.find("ZoneIndex");
613998fbe67SJosh Lehan if (findZoneIndex == zone.end())
614998fbe67SJosh Lehan {
615998fbe67SJosh Lehan continue;
616998fbe67SJosh Lehan }
617998fbe67SJosh Lehan
618998fbe67SJosh Lehan auto ptrZoneIndex = std::get_if<double>(&(findZoneIndex->second));
619998fbe67SJosh Lehan if (!ptrZoneIndex)
620998fbe67SJosh Lehan {
621998fbe67SJosh Lehan continue;
622998fbe67SJosh Lehan }
623998fbe67SJosh Lehan
624998fbe67SJosh Lehan auto desiredIndex = static_cast<int64_t>(*ptrZoneIndex);
625998fbe67SJosh Lehan auto grantedIndex = setZoneIndex(name, foundZones, desiredIndex);
626998fbe67SJosh Lehan std::cout << "Zone " << name << " is at ZoneIndex " << grantedIndex
627998fbe67SJosh Lehan << "\n";
628998fbe67SJosh Lehan }
629998fbe67SJosh Lehan }
630998fbe67SJosh Lehan
631998fbe67SJosh Lehan for (const auto& configuration : configurations)
632998fbe67SJosh Lehan {
633998fbe67SJosh Lehan auto findZone =
634998fbe67SJosh Lehan configuration.second.find(pidZoneConfigurationInterface);
635998fbe67SJosh Lehan if (findZone != configuration.second.end())
636998fbe67SJosh Lehan {
637998fbe67SJosh Lehan const auto& zone = findZone->second;
638998fbe67SJosh Lehan
639998fbe67SJosh Lehan const std::string& name = std::get<std::string>(zone.at("Name"));
640998fbe67SJosh Lehan
641998fbe67SJosh Lehan auto index = getZoneIndex(name, foundZones);
6428c3c51eeSJames Feist
643c54fbd88SPatrick Venture auto& details = zoneDetailsConfig[index];
644998fbe67SJosh Lehan
6453484bedaSJames Feist details.minThermalOutput = std::visit(VariantToDoubleVisitor(),
6463484bedaSJames Feist zone.at("MinThermalOutput"));
6479fe3a3c7Sykchiu
6489fe3a3c7Sykchiu int failsafepercent = 0;
6499fe3a3c7Sykchiu auto findFailSafe = zone.find("FailSafePercent");
6509fe3a3c7Sykchiu if (findFailSafe != zone.end())
6519fe3a3c7Sykchiu {
6529fe3a3c7Sykchiu failsafepercent = std::visit(VariantToDoubleVisitor(),
6531f802f5eSJames Feist zone.at("FailSafePercent"));
6549fe3a3c7Sykchiu }
6559fe3a3c7Sykchiu details.failsafePercent = failsafepercent;
6569f9a06aaSJosh Lehan
657239aa7d7SHarvey Wu getCycleTimeSetting(zone, index, "CycleIntervalTimeMS",
658239aa7d7SHarvey Wu details.cycleTime.cycleIntervalTimeMS);
659239aa7d7SHarvey Wu getCycleTimeSetting(zone, index, "UpdateThermalsTimeMS",
660239aa7d7SHarvey Wu details.cycleTime.updateThermalsTimeMS);
6619788963cSDelphine CC Chiu
6629788963cSDelphine CC Chiu bool accumulateSetPoint = false;
6639788963cSDelphine CC Chiu auto findAccSetPoint = zone.find("AccumulateSetPoint");
6649788963cSDelphine CC Chiu if (findAccSetPoint != zone.end())
6659788963cSDelphine CC Chiu {
6669788963cSDelphine CC Chiu accumulateSetPoint = std::get<bool>(findAccSetPoint->second);
6679788963cSDelphine CC Chiu }
6689788963cSDelphine CC Chiu details.accumulateSetPoint = accumulateSetPoint;
6697136a5aeSJames Feist }
6707136a5aeSJames Feist auto findBase = configuration.second.find(pidConfigurationInterface);
671f3b04fd3SJason Ling // loop through all the PID configurations and fill out a sensor config
67222c257abSJames Feist if (findBase != configuration.second.end())
6737136a5aeSJames Feist {
67422c257abSJames Feist const auto& base =
67522c257abSJames Feist configuration.second.at(pidConfigurationInterface);
6767c6d35d5Sykchiu const std::string pidName =
6777c6d35d5Sykchiu sensorNameToDbusName(std::get<std::string>(base.at("Name")));
678f3b04fd3SJason Ling const std::string pidClass =
679f3b04fd3SJason Ling std::get<std::string>(base.at("Class"));
6808c3c51eeSJames Feist const std::vector<std::string>& zones =
6811f802f5eSJames Feist std::get<std::vector<std::string>>(base.at("Zones"));
6828c3c51eeSJames Feist for (const std::string& zone : zones)
6838c3c51eeSJames Feist {
684998fbe67SJosh Lehan auto index = getZoneIndex(zone, foundZones);
685998fbe67SJosh Lehan
686f81f2886SJames Feist conf::PIDConf& conf = zoneConfig[index];
687f3b04fd3SJason Ling std::vector<std::string> inputSensorNames(
688f3b04fd3SJason Ling std::get<std::vector<std::string>>(base.at("Inputs")));
689f3b04fd3SJason Ling std::vector<std::string> outputSensorNames;
6903f0f7bc3SJosh Lehan std::vector<std::string> missingAcceptableSensorNames;
691*a552fe2fSChaul Ly std::vector<std::string> archivedInputSensorNames;
6923f0f7bc3SJosh Lehan
6933f0f7bc3SJosh Lehan auto findMissingAcceptable = base.find("MissingIsAcceptable");
6943f0f7bc3SJosh Lehan if (findMissingAcceptable != base.end())
6953f0f7bc3SJosh Lehan {
6963f0f7bc3SJosh Lehan missingAcceptableSensorNames =
6973f0f7bc3SJosh Lehan std::get<std::vector<std::string>>(
6983f0f7bc3SJosh Lehan findMissingAcceptable->second);
6993f0f7bc3SJosh Lehan }
70050fdfe39SJames Feist
701f3b04fd3SJason Ling // assumption: all fan pids must have at least one output
702f3b04fd3SJason Ling if (pidClass == "fan")
70350fdfe39SJames Feist {
704f3b04fd3SJason Ling outputSensorNames = std::get<std::vector<std::string>>(
705f3b04fd3SJason Ling getPIDAttribute(base, "Outputs"));
70650fdfe39SJames Feist }
7071738e2a2SJames Feist
7088f73ad76SAlex.Song bool unavailableAsFailed = true;
7098f73ad76SAlex.Song auto findUnavailableAsFailed =
7108f73ad76SAlex.Song base.find("InputUnavailableAsFailed");
7118f73ad76SAlex.Song if (findUnavailableAsFailed != base.end())
7128f73ad76SAlex.Song {
7138f73ad76SAlex.Song unavailableAsFailed =
7148f73ad76SAlex.Song std::get<bool>(findUnavailableAsFailed->second);
7158f73ad76SAlex.Song }
7168f73ad76SAlex.Song
717f3b04fd3SJason Ling std::vector<SensorInterfaceType> inputSensorInterfaces;
718f3b04fd3SJason Ling std::vector<SensorInterfaceType> outputSensorInterfaces;
7193f0f7bc3SJosh Lehan std::vector<SensorInterfaceType>
7203f0f7bc3SJosh Lehan missingAcceptableSensorInterfaces;
7213f0f7bc3SJosh Lehan
722f3b04fd3SJason Ling /* populate an interface list for different sensor direction
723f3b04fd3SJason Ling * types (input,output)
724f3b04fd3SJason Ling */
725f3b04fd3SJason Ling /* take the Inputs from the configuration and generate
726f3b04fd3SJason Ling * a list of dbus descriptors (path, interface).
727f3b04fd3SJason Ling * Mapping can be many-to-one since an element of Inputs can be
728f3b04fd3SJason Ling * a regex
729f3b04fd3SJason Ling */
730f3b04fd3SJason Ling for (const std::string& sensorName : inputSensorNames)
73150fdfe39SJames Feist {
732*a552fe2fSChaul Ly #ifndef HANDLE_MISSING_OBJECT_PATHS
733f3b04fd3SJason Ling findSensors(sensors, sensorNameToDbusName(sensorName),
734f3b04fd3SJason Ling inputSensorInterfaces);
735*a552fe2fSChaul Ly #else
736*a552fe2fSChaul Ly std::vector<std::pair<std::string, std::string>>
737*a552fe2fSChaul Ly sensorPathIfacePairs;
738*a552fe2fSChaul Ly auto found =
739*a552fe2fSChaul Ly findSensors(sensors, sensorNameToDbusName(sensorName),
740*a552fe2fSChaul Ly sensorPathIfacePairs);
741*a552fe2fSChaul Ly if (found)
742*a552fe2fSChaul Ly {
743*a552fe2fSChaul Ly inputSensorInterfaces.insert(
744*a552fe2fSChaul Ly inputSensorInterfaces.end(),
745*a552fe2fSChaul Ly sensorPathIfacePairs.begin(),
746*a552fe2fSChaul Ly sensorPathIfacePairs.end());
747*a552fe2fSChaul Ly }
748*a552fe2fSChaul Ly else if (pidClass != "fan")
749*a552fe2fSChaul Ly {
750*a552fe2fSChaul Ly if (std::find(missingAcceptableSensorNames.begin(),
751*a552fe2fSChaul Ly missingAcceptableSensorNames.end(),
752*a552fe2fSChaul Ly sensorName) ==
753*a552fe2fSChaul Ly missingAcceptableSensorNames.end())
754*a552fe2fSChaul Ly {
755*a552fe2fSChaul Ly std::cerr
756*a552fe2fSChaul Ly << "Pid controller: Missing a missing-unacceptable sensor from D-Bus "
757*a552fe2fSChaul Ly << sensorName << "\n";
758*a552fe2fSChaul Ly std::string inputSensorName =
759*a552fe2fSChaul Ly sensorNameToDbusName(sensorName);
760*a552fe2fSChaul Ly auto& config = sensorConfig[inputSensorName];
761*a552fe2fSChaul Ly archivedInputSensorNames.push_back(inputSensorName);
762*a552fe2fSChaul Ly config.type = pidClass;
763*a552fe2fSChaul Ly config.readPath =
764*a552fe2fSChaul Ly getSensorPath(config.type, inputSensorName);
765*a552fe2fSChaul Ly config.timeout = 0;
766*a552fe2fSChaul Ly config.ignoreDbusMinMax = true;
767*a552fe2fSChaul Ly config.unavailableAsFailed = unavailableAsFailed;
768*a552fe2fSChaul Ly }
769*a552fe2fSChaul Ly else
770*a552fe2fSChaul Ly {
771*a552fe2fSChaul Ly // When an input sensor is NOT on DBus, and it's in
772*a552fe2fSChaul Ly // the MissingIsAcceptable list. Ignore it and
773*a552fe2fSChaul Ly // continue with the next input sensor.
774*a552fe2fSChaul Ly std::cout
775*a552fe2fSChaul Ly << "Pid controller: Missing a missing-acceptable sensor from D-Bus "
776*a552fe2fSChaul Ly << sensorName << "\n";
777*a552fe2fSChaul Ly continue;
778*a552fe2fSChaul Ly }
779*a552fe2fSChaul Ly }
780*a552fe2fSChaul Ly #endif
781f3b04fd3SJason Ling }
782f3b04fd3SJason Ling for (const std::string& sensorName : outputSensorNames)
783f3b04fd3SJason Ling {
784f3b04fd3SJason Ling findSensors(sensors, sensorNameToDbusName(sensorName),
785f3b04fd3SJason Ling outputSensorInterfaces);
78650fdfe39SJames Feist }
7873f0f7bc3SJosh Lehan for (const std::string& sensorName :
7883f0f7bc3SJosh Lehan missingAcceptableSensorNames)
7893f0f7bc3SJosh Lehan {
7903f0f7bc3SJosh Lehan findSensors(sensors, sensorNameToDbusName(sensorName),
7913f0f7bc3SJosh Lehan missingAcceptableSensorInterfaces);
7923f0f7bc3SJosh Lehan }
7931738e2a2SJames Feist
794f3b04fd3SJason Ling for (const SensorInterfaceType& inputSensorInterface :
795f3b04fd3SJason Ling inputSensorInterfaces)
7961738e2a2SJames Feist {
797f3b04fd3SJason Ling const std::string& dbusInterface =
798f3b04fd3SJason Ling inputSensorInterface.second;
799f3b04fd3SJason Ling const std::string& inputSensorPath =
800f3b04fd3SJason Ling inputSensorInterface.first;
801fb82a87dSJosh Lehan
802fb82a87dSJosh Lehan // Setting timeout to 0 is intentional, as D-Bus passive
803fb82a87dSJosh Lehan // sensor updates are pushed in, not pulled by timer poll.
804fb82a87dSJosh Lehan // Setting ignoreDbusMinMax is intentional, as this
805fb82a87dSJosh Lehan // prevents normalization of values to [0.0, 1.0] range,
806fb82a87dSJosh Lehan // which would mess up the PID loop math.
807fb82a87dSJosh Lehan // All non-fan PID classes should be initialized this way.
808fb82a87dSJosh Lehan // As for why a fan should not use this code path, see
809fb82a87dSJosh Lehan // the ed1dafdf168def37c65bfb7a5efd18d9dbe04727 commit.
81023e22b90SJosh Lehan if ((pidClass == "temp") || (pidClass == "margin") ||
81123e22b90SJosh Lehan (pidClass == "power") || (pidClass == "powersum"))
812ed1dafdfSHarvey.Wu {
813f3b04fd3SJason Ling std::string inputSensorName =
814f3b04fd3SJason Ling getSensorNameFromPath(inputSensorPath);
815f3b04fd3SJason Ling auto& config = sensorConfig[inputSensorName];
816*a552fe2fSChaul Ly archivedInputSensorNames.push_back(inputSensorName);
817f3b04fd3SJason Ling config.type = pidClass;
818f3b04fd3SJason Ling config.readPath = inputSensorInterface.first;
8192642cb54SJames Feist config.timeout = 0;
8203433cb60SJames Feist config.ignoreDbusMinMax = true;
8218f73ad76SAlex.Song config.unavailableAsFailed = unavailableAsFailed;
82250fdfe39SJames Feist }
823fb82a87dSJosh Lehan
824f3b04fd3SJason Ling if (dbusInterface != sensorInterface)
825f3b04fd3SJason Ling {
826f3b04fd3SJason Ling /* all expected inputs in the configuration are expected
827f3b04fd3SJason Ling * to be sensor interfaces
828f3b04fd3SJason Ling */
829f3b04fd3SJason Ling throw std::runtime_error(
830f3b04fd3SJason Ling "sensor at dbus path [" + inputSensorPath +
831f3b04fd3SJason Ling "] has an interface [" + dbusInterface +
832f3b04fd3SJason Ling "] that does not match the expected interface of " +
833f3b04fd3SJason Ling sensorInterface);
83450fdfe39SJames Feist }
83550fdfe39SJames Feist }
8361738e2a2SJames Feist
8373f0f7bc3SJosh Lehan // MissingIsAcceptable same postprocessing as Inputs
8383f0f7bc3SJosh Lehan missingAcceptableSensorNames.clear();
8393f0f7bc3SJosh Lehan for (const SensorInterfaceType&
8403f0f7bc3SJosh Lehan missingAcceptableSensorInterface :
8413f0f7bc3SJosh Lehan missingAcceptableSensorInterfaces)
8423f0f7bc3SJosh Lehan {
8433f0f7bc3SJosh Lehan const std::string& dbusInterface =
8443f0f7bc3SJosh Lehan missingAcceptableSensorInterface.second;
8453f0f7bc3SJosh Lehan const std::string& missingAcceptableSensorPath =
8463f0f7bc3SJosh Lehan missingAcceptableSensorInterface.first;
8473f0f7bc3SJosh Lehan
8483f0f7bc3SJosh Lehan std::string missingAcceptableSensorName =
8493f0f7bc3SJosh Lehan getSensorNameFromPath(missingAcceptableSensorPath);
8503f0f7bc3SJosh Lehan missingAcceptableSensorNames.push_back(
8513f0f7bc3SJosh Lehan missingAcceptableSensorName);
8523f0f7bc3SJosh Lehan
8533f0f7bc3SJosh Lehan if (dbusInterface != sensorInterface)
8543f0f7bc3SJosh Lehan {
8553f0f7bc3SJosh Lehan /* MissingIsAcceptable same error checking as Inputs
8563f0f7bc3SJosh Lehan */
8573f0f7bc3SJosh Lehan throw std::runtime_error(
8583f0f7bc3SJosh Lehan "sensor at dbus path [" +
8593f0f7bc3SJosh Lehan missingAcceptableSensorPath +
8603f0f7bc3SJosh Lehan "] has an interface [" + dbusInterface +
8613f0f7bc3SJosh Lehan "] that does not match the expected interface of " +
8623f0f7bc3SJosh Lehan sensorInterface);
8633f0f7bc3SJosh Lehan }
8643f0f7bc3SJosh Lehan }
8653f0f7bc3SJosh Lehan
866f3b04fd3SJason Ling /* fan pids need to pair up tach sensors with their pwm
867f3b04fd3SJason Ling * counterparts
868f3b04fd3SJason Ling */
869f3b04fd3SJason Ling if (pidClass == "fan")
870f3b04fd3SJason Ling {
871f3b04fd3SJason Ling /* If a PID is a fan there should be either
872f3b04fd3SJason Ling * (1) one output(pwm) per input(tach)
873f3b04fd3SJason Ling * OR
874f3b04fd3SJason Ling * (2) one putput(pwm) for all inputs(tach)
875f3b04fd3SJason Ling * everything else indicates a bad configuration.
876f3b04fd3SJason Ling */
877f3b04fd3SJason Ling bool singlePwm = false;
878f3b04fd3SJason Ling if (outputSensorInterfaces.size() == 1)
879f3b04fd3SJason Ling {
880f3b04fd3SJason Ling /* one pwm, set write paths for all fan sensors to it */
881f3b04fd3SJason Ling singlePwm = true;
882f3b04fd3SJason Ling }
883f3b04fd3SJason Ling else if (inputSensorInterfaces.size() ==
884f3b04fd3SJason Ling outputSensorInterfaces.size())
885f3b04fd3SJason Ling {
886f3b04fd3SJason Ling /* one to one mapping, each fan sensor gets its own pwm
887f3b04fd3SJason Ling * control */
888f3b04fd3SJason Ling singlePwm = false;
889f3b04fd3SJason Ling }
890f3b04fd3SJason Ling else
891f3b04fd3SJason Ling {
892f3b04fd3SJason Ling throw std::runtime_error(
893f3b04fd3SJason Ling "fan PID has invalid number of Outputs");
894f3b04fd3SJason Ling }
895f3b04fd3SJason Ling std::string fanSensorName;
896f3b04fd3SJason Ling std::string pwmPath;
897f3b04fd3SJason Ling std::string pwmInterface;
898ed1dafdfSHarvey.Wu std::string pwmSensorName;
899f3b04fd3SJason Ling if (singlePwm)
900f3b04fd3SJason Ling {
901f3b04fd3SJason Ling /* if just a single output(pwm) is provided then use
902f3b04fd3SJason Ling * that pwm control path for all the fan sensor write
903f3b04fd3SJason Ling * path configs
904f3b04fd3SJason Ling */
905f3b04fd3SJason Ling pwmPath = outputSensorInterfaces.at(0).first;
906f3b04fd3SJason Ling pwmInterface = outputSensorInterfaces.at(0).second;
907f3b04fd3SJason Ling }
908f3b04fd3SJason Ling for (uint32_t idx = 0; idx < inputSensorInterfaces.size();
909f3b04fd3SJason Ling idx++)
910f3b04fd3SJason Ling {
911f3b04fd3SJason Ling if (!singlePwm)
912f3b04fd3SJason Ling {
913f3b04fd3SJason Ling pwmPath = outputSensorInterfaces.at(idx).first;
914f3b04fd3SJason Ling pwmInterface =
915f3b04fd3SJason Ling outputSensorInterfaces.at(idx).second;
916f3b04fd3SJason Ling }
9170911bfe7SPatrick Venture if (defaultPwmInterface != pwmInterface)
918f3b04fd3SJason Ling {
919f3b04fd3SJason Ling throw std::runtime_error(
920f3b04fd3SJason Ling "fan pwm control at dbus path [" + pwmPath +
921f3b04fd3SJason Ling "] has an interface [" + pwmInterface +
922f3b04fd3SJason Ling "] that does not match the expected interface "
923f3b04fd3SJason Ling "of " +
9240911bfe7SPatrick Venture defaultPwmInterface);
925f3b04fd3SJason Ling }
926f3b04fd3SJason Ling const std::string& fanPath =
927f3b04fd3SJason Ling inputSensorInterfaces.at(idx).first;
928f3b04fd3SJason Ling fanSensorName = getSensorNameFromPath(fanPath);
929ed1dafdfSHarvey.Wu pwmSensorName = getSensorNameFromPath(pwmPath);
930ed1dafdfSHarvey.Wu std::string fanPwmIndex = fanSensorName + pwmSensorName;
931*a552fe2fSChaul Ly archivedInputSensorNames.push_back(fanPwmIndex);
932ed1dafdfSHarvey.Wu auto& fanConfig = sensorConfig[fanPwmIndex];
933ed1dafdfSHarvey.Wu fanConfig.type = pidClass;
934ed1dafdfSHarvey.Wu fanConfig.readPath = fanPath;
935f3b04fd3SJason Ling fanConfig.writePath = pwmPath;
93650fdfe39SJames Feist // todo: un-hardcode this if there are fans with
93750fdfe39SJames Feist // different ranges
938f3b04fd3SJason Ling fanConfig.max = 255;
939f3b04fd3SJason Ling fanConfig.min = 0;
94050fdfe39SJames Feist }
94150fdfe39SJames Feist }
94211d243dfSJames Feist // if the sensors aren't available in the current state, don't
94311d243dfSJames Feist // add them to the configuration.
944*a552fe2fSChaul Ly if (archivedInputSensorNames.empty())
94511d243dfSJames Feist {
94611d243dfSJames Feist continue;
94711d243dfSJames Feist }
94811d243dfSJames Feist
9495ec20270SJames Feist std::string offsetType;
9505ec20270SJames Feist
9515ec20270SJames Feist // SetPointOffset is a threshold value to pull from the sensor
9525ec20270SJames Feist // to apply an offset. For upper thresholds this means the
9535ec20270SJames Feist // setpoint is usually negative.
9545ec20270SJames Feist auto findSetpointOffset = base.find("SetPointOffset");
9555ec20270SJames Feist if (findSetpointOffset != base.end())
9565ec20270SJames Feist {
9575ec20270SJames Feist offsetType =
9585ec20270SJames Feist std::get<std::string>(findSetpointOffset->second);
9595ec20270SJames Feist if (std::find(thresholds::types.begin(),
960bd63bcacSPatrick Williams thresholds::types.end(), offsetType) ==
961bd63bcacSPatrick Williams thresholds::types.end())
9625ec20270SJames Feist {
963bd63bcacSPatrick Williams throw std::runtime_error(
964bd63bcacSPatrick Williams "Unsupported type: " + offsetType);
9655ec20270SJames Feist }
9665ec20270SJames Feist }
9675ec20270SJames Feist
96831058fd3SJosh Lehan std::vector<double> inputTempToMargin;
96931058fd3SJosh Lehan
97031058fd3SJosh Lehan auto findTempToMargin = base.find("TempToMargin");
97131058fd3SJosh Lehan if (findTempToMargin != base.end())
97231058fd3SJosh Lehan {
97331058fd3SJosh Lehan inputTempToMargin =
97431058fd3SJosh Lehan std::get<std::vector<double>>(findTempToMargin->second);
97531058fd3SJosh Lehan }
97631058fd3SJosh Lehan
97731058fd3SJosh Lehan std::vector<pid_control::conf::SensorInput> sensorInputs =
978*a552fe2fSChaul Ly spliceInputs(archivedInputSensorNames, inputTempToMargin,
9793f0f7bc3SJosh Lehan missingAcceptableSensorNames);
98031058fd3SJosh Lehan
9815ec20270SJames Feist if (offsetType.empty())
9825ec20270SJames Feist {
9837c6d35d5Sykchiu conf::ControllerInfo& info = conf[pidName];
98431058fd3SJosh Lehan info.inputs = std::move(sensorInputs);
9857382318fSPatrick Venture populatePidInfo(bus, base, info, nullptr, sensorConfig);
9867136a5aeSJames Feist }
9877136a5aeSJames Feist else
9887136a5aeSJames Feist {
9895ec20270SJames Feist // we have to split up the inputs, as in practice t-control
9905ec20270SJames Feist // values will differ, making setpoints differ
99131058fd3SJosh Lehan for (const pid_control::conf::SensorInput& input :
99231058fd3SJosh Lehan sensorInputs)
993572c43daSJames Feist {
99431058fd3SJosh Lehan conf::ControllerInfo& info = conf[input.name];
9955ec20270SJames Feist info.inputs.emplace_back(input);
9967382318fSPatrick Venture populatePidInfo(bus, base, info, &offsetType,
9977382318fSPatrick Venture sensorConfig);
998572c43daSJames Feist }
999572c43daSJames Feist }
10007136a5aeSJames Feist }
10018c3c51eeSJames Feist }
100222c257abSJames Feist auto findStepwise =
100322c257abSJames Feist configuration.second.find(stepwiseConfigurationInterface);
100422c257abSJames Feist if (findStepwise != configuration.second.end())
100522c257abSJames Feist {
100622c257abSJames Feist const auto& base = findStepwise->second;
10077c6d35d5Sykchiu const std::string pidName =
10087c6d35d5Sykchiu sensorNameToDbusName(std::get<std::string>(base.at("Name")));
100922c257abSJames Feist const std::vector<std::string>& zones =
10101f802f5eSJames Feist std::get<std::vector<std::string>>(base.at("Zones"));
101122c257abSJames Feist for (const std::string& zone : zones)
101222c257abSJames Feist {
1013998fbe67SJosh Lehan auto index = getZoneIndex(zone, foundZones);
1014998fbe67SJosh Lehan
1015f81f2886SJames Feist conf::PIDConf& conf = zoneConfig[index];
101650fdfe39SJames Feist
101750fdfe39SJames Feist std::vector<std::string> inputs;
10183f0f7bc3SJosh Lehan std::vector<std::string> missingAcceptableSensors;
10193f0f7bc3SJosh Lehan std::vector<std::string> missingAcceptableSensorNames;
102050fdfe39SJames Feist std::vector<std::string> sensorNames =
10211f802f5eSJames Feist std::get<std::vector<std::string>>(base.at("Inputs"));
102250fdfe39SJames Feist
10233f0f7bc3SJosh Lehan auto findMissingAcceptable = base.find("MissingIsAcceptable");
10243f0f7bc3SJosh Lehan if (findMissingAcceptable != base.end())
10253f0f7bc3SJosh Lehan {
10263f0f7bc3SJosh Lehan missingAcceptableSensorNames =
10273f0f7bc3SJosh Lehan std::get<std::vector<std::string>>(
10283f0f7bc3SJosh Lehan findMissingAcceptable->second);
10293f0f7bc3SJosh Lehan }
10303f0f7bc3SJosh Lehan
10318f73ad76SAlex.Song bool unavailableAsFailed = true;
10328f73ad76SAlex.Song auto findUnavailableAsFailed =
10338f73ad76SAlex.Song base.find("InputUnavailableAsFailed");
10348f73ad76SAlex.Song if (findUnavailableAsFailed != base.end())
10358f73ad76SAlex.Song {
10368f73ad76SAlex.Song unavailableAsFailed =
10378f73ad76SAlex.Song std::get<bool>(findUnavailableAsFailed->second);
10388f73ad76SAlex.Song }
10398f73ad76SAlex.Song
10401738e2a2SJames Feist bool sensorFound = false;
104150fdfe39SJames Feist for (const std::string& sensorName : sensorNames)
104250fdfe39SJames Feist {
10431738e2a2SJames Feist std::vector<std::pair<std::string, std::string>>
10441738e2a2SJames Feist sensorPathIfacePairs;
1045f3b04fd3SJason Ling if (!findSensors(sensors, sensorNameToDbusName(sensorName),
1046f3b04fd3SJason Ling sensorPathIfacePairs))
104750fdfe39SJames Feist {
1048*a552fe2fSChaul Ly #ifndef HANDLE_MISSING_OBJECT_PATHS
104950fdfe39SJames Feist break;
1050*a552fe2fSChaul Ly #else
1051*a552fe2fSChaul Ly if (std::find(missingAcceptableSensorNames.begin(),
1052*a552fe2fSChaul Ly missingAcceptableSensorNames.end(),
1053*a552fe2fSChaul Ly sensorName) ==
1054*a552fe2fSChaul Ly missingAcceptableSensorNames.end())
10551738e2a2SJames Feist {
1056*a552fe2fSChaul Ly // When an input sensor is NOT on DBus, and it's NOT
1057*a552fe2fSChaul Ly // in the MissingIsAcceptable list. Build it as a
1058*a552fe2fSChaul Ly // failed sensor with default information (temp
1059*a552fe2fSChaul Ly // sensor path, temp type, ...)
1060*a552fe2fSChaul Ly std::cerr
1061*a552fe2fSChaul Ly << "Stepwise controller: Missing a missing-unacceptable sensor from D-Bus "
1062*a552fe2fSChaul Ly << sensorName << "\n";
10631738e2a2SJames Feist std::string shortName =
1064*a552fe2fSChaul Ly sensorNameToDbusName(sensorName);
1065*a552fe2fSChaul Ly
1066*a552fe2fSChaul Ly inputs.push_back(shortName);
1067*a552fe2fSChaul Ly auto& config = sensorConfig[shortName];
1068*a552fe2fSChaul Ly config.type = "temp";
1069*a552fe2fSChaul Ly config.readPath =
1070*a552fe2fSChaul Ly getSensorPath(config.type, shortName);
1071*a552fe2fSChaul Ly config.ignoreDbusMinMax = true;
1072*a552fe2fSChaul Ly config.unavailableAsFailed = unavailableAsFailed;
1073*a552fe2fSChaul Ly // todo: maybe un-hardcode this if we run into
1074*a552fe2fSChaul Ly // slower timeouts with sensors
1075*a552fe2fSChaul Ly
1076*a552fe2fSChaul Ly config.timeout = 0;
1077*a552fe2fSChaul Ly sensorFound = true;
1078*a552fe2fSChaul Ly }
1079*a552fe2fSChaul Ly else
1080*a552fe2fSChaul Ly {
1081*a552fe2fSChaul Ly // When an input sensor is NOT on DBus, and it's in
1082*a552fe2fSChaul Ly // the MissingIsAcceptable list. Ignore it and
1083*a552fe2fSChaul Ly // continue with the next input sensor.
1084*a552fe2fSChaul Ly std::cout
1085*a552fe2fSChaul Ly << "Stepwise controller: Missing a missing-acceptable sensor from D-Bus "
1086*a552fe2fSChaul Ly << sensorName << "\n";
1087*a552fe2fSChaul Ly continue;
1088*a552fe2fSChaul Ly }
1089*a552fe2fSChaul Ly #endif
1090*a552fe2fSChaul Ly }
1091*a552fe2fSChaul Ly else
1092*a552fe2fSChaul Ly {
1093*a552fe2fSChaul Ly for (const auto& sensorPathIfacePair :
1094*a552fe2fSChaul Ly sensorPathIfacePairs)
1095*a552fe2fSChaul Ly {
1096*a552fe2fSChaul Ly std::string shortName = getSensorNameFromPath(
1097*a552fe2fSChaul Ly sensorPathIfacePair.first);
10981738e2a2SJames Feist
10991738e2a2SJames Feist inputs.push_back(shortName);
11001738e2a2SJames Feist auto& config = sensorConfig[shortName];
110169c51061SPatrick Venture config.readPath = sensorPathIfacePair.first;
110250fdfe39SJames Feist config.type = "temp";
11033660b388SJames Feist config.ignoreDbusMinMax = true;
11048f73ad76SAlex.Song config.unavailableAsFailed = unavailableAsFailed;
1105*a552fe2fSChaul Ly // todo: maybe un-hardcode this if we run into
1106*a552fe2fSChaul Ly // slower timeouts with sensors
110750fdfe39SJames Feist
11082642cb54SJames Feist config.timeout = 0;
11091738e2a2SJames Feist sensorFound = true;
11101738e2a2SJames Feist }
111150fdfe39SJames Feist }
1112*a552fe2fSChaul Ly }
111350fdfe39SJames Feist if (!sensorFound)
111450fdfe39SJames Feist {
111550fdfe39SJames Feist continue;
111650fdfe39SJames Feist }
11173f0f7bc3SJosh Lehan
11183f0f7bc3SJosh Lehan // MissingIsAcceptable same postprocessing as Inputs
11193f0f7bc3SJosh Lehan for (const std::string& missingAcceptableSensorName :
11203f0f7bc3SJosh Lehan missingAcceptableSensorNames)
11213f0f7bc3SJosh Lehan {
11223f0f7bc3SJosh Lehan std::vector<std::pair<std::string, std::string>>
11233f0f7bc3SJosh Lehan sensorPathIfacePairs;
11243f0f7bc3SJosh Lehan if (!findSensors(
11253f0f7bc3SJosh Lehan sensors,
11263f0f7bc3SJosh Lehan sensorNameToDbusName(missingAcceptableSensorName),
11273f0f7bc3SJosh Lehan sensorPathIfacePairs))
11283f0f7bc3SJosh Lehan {
1129*a552fe2fSChaul Ly #ifndef HANDLE_MISSING_OBJECT_PATHS
11303f0f7bc3SJosh Lehan break;
1131*a552fe2fSChaul Ly #else
1132*a552fe2fSChaul Ly // When a sensor in the MissingIsAcceptable list is NOT
1133*a552fe2fSChaul Ly // on DBus and it still reaches here, which contradicts
1134*a552fe2fSChaul Ly // to what we did in the Input sensor building step.
1135*a552fe2fSChaul Ly // Continue.
1136*a552fe2fSChaul Ly continue;
1137*a552fe2fSChaul Ly #endif
11383f0f7bc3SJosh Lehan }
11393f0f7bc3SJosh Lehan
11403f0f7bc3SJosh Lehan for (const auto& sensorPathIfacePair : sensorPathIfacePairs)
11413f0f7bc3SJosh Lehan {
11423f0f7bc3SJosh Lehan std::string shortName =
11433f0f7bc3SJosh Lehan getSensorNameFromPath(sensorPathIfacePair.first);
11443f0f7bc3SJosh Lehan
11453f0f7bc3SJosh Lehan missingAcceptableSensors.push_back(shortName);
11463f0f7bc3SJosh Lehan }
11473f0f7bc3SJosh Lehan }
11483f0f7bc3SJosh Lehan
11497c6d35d5Sykchiu conf::ControllerInfo& info = conf[pidName];
115031058fd3SJosh Lehan
115131058fd3SJosh Lehan std::vector<double> inputTempToMargin;
115231058fd3SJosh Lehan
115331058fd3SJosh Lehan auto findTempToMargin = base.find("TempToMargin");
115431058fd3SJosh Lehan if (findTempToMargin != base.end())
115531058fd3SJosh Lehan {
115631058fd3SJosh Lehan inputTempToMargin =
115731058fd3SJosh Lehan std::get<std::vector<double>>(findTempToMargin->second);
115831058fd3SJosh Lehan }
115931058fd3SJosh Lehan
11603f0f7bc3SJosh Lehan info.inputs = spliceInputs(inputs, inputTempToMargin,
11613f0f7bc3SJosh Lehan missingAcceptableSensors);
116250fdfe39SJames Feist
116322c257abSJames Feist info.type = "stepwise";
116422c257abSJames Feist info.stepwiseInfo.ts = 1.0; // currently unused
11653dfaafdaSJames Feist info.stepwiseInfo.positiveHysteresis = 0.0;
11663dfaafdaSJames Feist info.stepwiseInfo.negativeHysteresis = 0.0;
1167608304daSJames Feist std::string subtype = std::get<std::string>(base.at("Class"));
1168608304daSJames Feist
1169608304daSJames Feist info.stepwiseInfo.isCeiling = (subtype == "Ceiling");
11703dfaafdaSJames Feist auto findPosHyst = base.find("PositiveHysteresis");
11713dfaafdaSJames Feist auto findNegHyst = base.find("NegativeHysteresis");
11723dfaafdaSJames Feist if (findPosHyst != base.end())
11733dfaafdaSJames Feist {
11741f802f5eSJames Feist info.stepwiseInfo.positiveHysteresis = std::visit(
1175208abce8SJames Feist VariantToDoubleVisitor(), findPosHyst->second);
11763dfaafdaSJames Feist }
11773dfaafdaSJames Feist if (findNegHyst != base.end())
11783dfaafdaSJames Feist {
11795782ab81SJames Feist info.stepwiseInfo.negativeHysteresis = std::visit(
1180208abce8SJames Feist VariantToDoubleVisitor(), findNegHyst->second);
11813dfaafdaSJames Feist }
118222c257abSJames Feist std::vector<double> readings =
11831f802f5eSJames Feist std::get<std::vector<double>>(base.at("Reading"));
118422c257abSJames Feist if (readings.size() > ec::maxStepwisePoints)
118522c257abSJames Feist {
118622c257abSJames Feist throw std::invalid_argument("Too many stepwise points.");
118722c257abSJames Feist }
118822c257abSJames Feist if (readings.empty())
118922c257abSJames Feist {
119022c257abSJames Feist throw std::invalid_argument(
119122c257abSJames Feist "Must have one stepwise point.");
119222c257abSJames Feist }
119322c257abSJames Feist std::copy(readings.begin(), readings.end(),
119422c257abSJames Feist info.stepwiseInfo.reading);
119522c257abSJames Feist if (readings.size() < ec::maxStepwisePoints)
119622c257abSJames Feist {
119722c257abSJames Feist info.stepwiseInfo.reading[readings.size()] =
11985f59c0fdSPatrick Venture std::numeric_limits<double>::quiet_NaN();
119922c257abSJames Feist }
120022c257abSJames Feist std::vector<double> outputs =
12011f802f5eSJames Feist std::get<std::vector<double>>(base.at("Output"));
120222c257abSJames Feist if (readings.size() != outputs.size())
120322c257abSJames Feist {
120422c257abSJames Feist throw std::invalid_argument(
120522c257abSJames Feist "Outputs size must match readings");
120622c257abSJames Feist }
120722c257abSJames Feist std::copy(outputs.begin(), outputs.end(),
120822c257abSJames Feist info.stepwiseInfo.output);
120922c257abSJames Feist if (outputs.size() < ec::maxStepwisePoints)
121022c257abSJames Feist {
121122c257abSJames Feist info.stepwiseInfo.output[outputs.size()] =
12125f59c0fdSPatrick Venture std::numeric_limits<double>::quiet_NaN();
121322c257abSJames Feist }
121422c257abSJames Feist }
121522c257abSJames Feist }
121622c257abSJames Feist }
121739199b4dSPatrick Venture if constexpr (pid_control::conf::DEBUG)
12187136a5aeSJames Feist {
121939199b4dSPatrick Venture debugPrint(sensorConfig, zoneConfig, zoneDetailsConfig);
12207136a5aeSJames Feist }
1221c959c429SJames Feist if (zoneConfig.empty() || zoneDetailsConfig.empty())
122250fdfe39SJames Feist {
12231fe08952SJames Feist std::cerr
12241fe08952SJames Feist << "No fan zones, application pausing until new configuration\n";
12251fe08952SJames Feist return false;
122650fdfe39SJames Feist }
12271fe08952SJames Feist return true;
12287136a5aeSJames Feist }
1229a076487aSPatrick Venture
12307136a5aeSJames Feist } // namespace dbus_configuration
1231a076487aSPatrick Venture } // namespace pid_control
1232