blob: bffead96b73f0e5ae1fffeed6bad2b71d7511e2c [file] [log] [blame]
James Feist6714a252018-09-10 15:26:18 -07001/*
2// Copyright (c) 2018 Intel Corporation
3//
4// Licensed under the Apache License, Version 2.0 (the "License");
5// you may not use this file except in compliance with the License.
6// You may obtain a copy of the License at
7//
8// http://www.apache.org/licenses/LICENSE-2.0
9//
10// Unless required by applicable law or agreed to in writing, software
11// distributed under the License is distributed on an "AS IS" BASIS,
12// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13// See the License for the specific language governing permissions and
14// limitations under the License.
15*/
Andrew Jefferye73bd0a2023-01-25 10:39:57 +103016
17#include "PwmSensor.hpp"
18
Ed Tanouseacbfdd2024-04-04 12:00:24 -070019#include "SensorPaths.hpp"
Andrew Jefferye73bd0a2023-01-25 10:39:57 +103020#include "Utils.hpp"
Ed Tanouseacbfdd2024-04-04 12:00:24 -070021#include "sensor.hpp"
Andrew Jefferye73bd0a2023-01-25 10:39:57 +103022
Ed Tanouseacbfdd2024-04-04 12:00:24 -070023#include <sdbusplus/asio/connection.hpp>
James Feist38fb5982020-05-28 10:09:54 -070024#include <sdbusplus/asio/object_server.hpp>
25
Ed Tanouseacbfdd2024-04-04 12:00:24 -070026#include <cmath>
27#include <cstdint>
James Feist6714a252018-09-10 15:26:18 -070028#include <fstream>
29#include <iostream>
Ed Tanouseacbfdd2024-04-04 12:00:24 -070030#include <memory>
Patrick Venture96e97db2019-10-31 13:44:38 -070031#include <stdexcept>
32#include <string>
James Feist6714a252018-09-10 15:26:18 -070033
Josh Lehan6e6ac0f2020-12-30 22:32:57 -080034static constexpr double sysPwmMax = 255.0;
35static constexpr double psuPwmMax = 100.0;
James Feist3a07f552019-08-27 13:34:54 -070036static constexpr double defaultPwm = 30.0;
Josh Lehan6e6ac0f2020-12-30 22:32:57 -080037static constexpr double targetIfaceMax = sysPwmMax;
James Feist6714a252018-09-10 15:26:18 -070038
Delphine CC Chiub0dff222023-09-28 10:19:00 +080039PwmSensor::PwmSensor(const std::string& pwmname, const std::string& sysPath,
AppaRao Pulid9d8caf2020-02-27 20:56:59 +053040 std::shared_ptr<sdbusplus::asio::connection>& conn,
James Feist82bac4c2019-03-11 11:16:53 -070041 sdbusplus::asio::object_server& objectServer,
AppaRao Pulid9d8caf2020-02-27 20:56:59 +053042 const std::string& sensorConfiguration,
Jie Yang3291b9c2021-07-29 14:46:51 -070043 const std::string& sensorType, bool isValueMutable) :
Patrick Williams2aaf7172024-08-16 15:20:40 -040044 sysPath(sysPath), objectServer(objectServer),
45 name(sensor_paths::escapePathForDbus(pwmname))
James Feist6714a252018-09-10 15:26:18 -070046{
James Feist6714a252018-09-10 15:26:18 -070047 // add interface under sensor and Control.FanPwm as Control is used
48 // in obmc project, also add sensor so it can be viewed as a sensor
49 sensorInterface = objectServer.add_interface(
50 "/xyz/openbmc_project/sensors/fan_pwm/" + name,
51 "xyz.openbmc_project.Sensor.Value");
52 uint32_t pwmValue = getValue(false);
Kuiying Wang105a1972020-08-28 19:07:53 +080053 if (sensorType == "PSU")
54 {
55 pwmMax = psuPwmMax;
56 }
57 else
58 {
59 pwmMax = sysPwmMax;
60 }
61
Ed Tanous2049bd22022-07-09 07:20:26 -070062 if (pwmValue == 0U)
James Feist3a07f552019-08-27 13:34:54 -070063 {
64 // default pwm to non 0
Josh Lehan6e6ac0f2020-12-30 22:32:57 -080065 pwmValue = static_cast<uint32_t>(pwmMax * (defaultPwm / 100.0));
James Feist3a07f552019-08-27 13:34:54 -070066 setValue(pwmValue);
67 }
James Feistb6c0b912019-07-09 12:21:44 -070068 double fValue = 100.0 * (static_cast<double>(pwmValue) / pwmMax);
James Feist6714a252018-09-10 15:26:18 -070069 sensorInterface->register_property(
70 "Value", fValue,
71 [this](const double& req, double& resp) {
Patrick Williams2aaf7172024-08-16 15:20:40 -040072 if (!std::isfinite(req))
73 {
74 // Reject attempted change, if to NaN or other non-numeric
75 return -1;
76 }
77 if (req > 100.0 || req < 0.0)
78 {
79 // TODO(): It does not seem desirable to halt daemon here,
80 // probably should just reject the change, continue running?
81 throw std::runtime_error("Value out of range");
82 return -1;
83 }
Josh Lehan6e6ac0f2020-12-30 22:32:57 -080084
Patrick Williams2aaf7172024-08-16 15:20:40 -040085 double reqValue = (req / 100.0) * pwmMax;
86 double respValue = (resp / 100.0) * pwmMax;
87 auto reqInt = static_cast<uint32_t>(std::round(reqValue));
88 auto respInt = static_cast<uint32_t>(std::round(respValue));
89 // Avoid floating-point equality, compare as integers
90 if (reqInt == respInt)
91 {
92 return 1;
93 }
94 setValue(reqInt);
95 resp = req;
Ed Tanousbb679322022-05-16 16:10:00 -070096
Ed Tanousbb679322022-05-16 16:10:00 -070097 controlInterface->signal_property("Target");
Patrick Williams2aaf7172024-08-16 15:20:40 -040098
99 return 1;
100 },
101 [this](double& curVal) {
102 double currScaled = (curVal / 100.0) * pwmMax;
103 auto currInt = static_cast<uint32_t>(std::round(currScaled));
104 auto getInt = getValue();
105 // Avoid floating-point equality, compare as integers
106 if (currInt != getInt)
107 {
108 double getScaled =
109 100.0 * (static_cast<double>(getInt) / pwmMax);
110 curVal = getScaled;
111 controlInterface->signal_property("Target");
112 sensorInterface->signal_property("Value");
113 }
114 return curVal;
115 });
James Feist6714a252018-09-10 15:26:18 -0700116 // pwm sensor interface is in percent
David Wangead7e922022-12-09 09:30:30 +0800117 sensorInterface->register_property("MaxValue", static_cast<double>(100));
118 sensorInterface->register_property("MinValue", static_cast<double>(0));
Zev Weiss6b6891c2021-04-22 02:46:21 -0500119 sensorInterface->register_property("Unit", sensor_paths::unitPercent);
James Feist6714a252018-09-10 15:26:18 -0700120
121 controlInterface = objectServer.add_interface(
122 "/xyz/openbmc_project/control/fanpwm/" + name,
123 "xyz.openbmc_project.Control.FanPwm");
124 controlInterface->register_property(
125 "Target", static_cast<uint64_t>(pwmValue),
126 [this](const uint64_t& req, uint64_t& resp) {
Patrick Williams2aaf7172024-08-16 15:20:40 -0400127 if (req > static_cast<uint64_t>(targetIfaceMax))
128 {
129 throw std::runtime_error("Value out of range");
130 return -1;
131 }
132 if (req == resp)
133 {
134 return 1;
135 }
136 auto scaledValue = static_cast<double>(req) / targetIfaceMax;
137 auto roundValue = std::round(scaledValue * pwmMax);
138 setValue(static_cast<uint32_t>(roundValue));
139 resp = req;
Ed Tanousbb679322022-05-16 16:10:00 -0700140
Ed Tanousbb679322022-05-16 16:10:00 -0700141 sensorInterface->signal_property("Value");
Patrick Williams2aaf7172024-08-16 15:20:40 -0400142
143 return 1;
144 },
145 [this](uint64_t& curVal) {
146 auto getInt = getValue();
147 auto scaledValue = static_cast<double>(getInt) / pwmMax;
148 auto roundValue = std::round(scaledValue * targetIfaceMax);
149 auto value = static_cast<uint64_t>(roundValue);
150 if (curVal != value)
151 {
152 curVal = value;
153 controlInterface->signal_property("Target");
154 sensorInterface->signal_property("Value");
155 }
156 return curVal;
157 });
Jie Yang3291b9c2021-07-29 14:46:51 -0700158
James Feist6714a252018-09-10 15:26:18 -0700159 sensorInterface->initialize();
160 controlInterface->initialize();
James Feist82bac4c2019-03-11 11:16:53 -0700161
Jie Yang3291b9c2021-07-29 14:46:51 -0700162 if (isValueMutable)
163 {
164 valueMutabilityInterface =
165 std::make_shared<sdbusplus::asio::dbus_interface>(
166 conn, sensorInterface->get_object_path(),
167 valueMutabilityInterfaceName);
168 valueMutabilityInterface->register_property("Mutable", true);
169 if (!valueMutabilityInterface->initialize())
170 {
171 std::cerr
172 << "error initializing sensor value mutability interface\n";
173 valueMutabilityInterface = nullptr;
174 }
175 }
176
James Feist82bac4c2019-03-11 11:16:53 -0700177 association = objectServer.add_interface(
James Feist2adc95c2019-09-30 14:55:28 -0700178 "/xyz/openbmc_project/sensors/fan_pwm/" + name, association::interface);
AppaRao Pulid9d8caf2020-02-27 20:56:59 +0530179
180 // PowerSupply sensors should be associated with chassis board path
181 // and inventory along with psu object.
182 if (sensorType == "PSU")
183 {
184 createInventoryAssoc(conn, association, sensorConfiguration);
185 }
186 else
187 {
188 createAssociation(association, sensorConfiguration);
189 }
James Feist6714a252018-09-10 15:26:18 -0700190}
191PwmSensor::~PwmSensor()
192{
193 objectServer.remove_interface(sensorInterface);
194 objectServer.remove_interface(controlInterface);
AppaRao Pulid9d8caf2020-02-27 20:56:59 +0530195 objectServer.remove_interface(association);
James Feist6714a252018-09-10 15:26:18 -0700196}
197
198void PwmSensor::setValue(uint32_t value)
199{
200 std::ofstream ref(sysPath);
201 if (!ref.good())
202 {
203 throw std::runtime_error("Bad Write File");
James Feist6714a252018-09-10 15:26:18 -0700204 }
205 ref << value;
206}
207
208// on success returns pwm, on failure throws except on initialization, where it
209// prints an error and returns 0
210uint32_t PwmSensor::getValue(bool errThrow)
211{
212 std::ifstream ref(sysPath);
213 if (!ref.good())
214 {
Yang Chene1dcb5c2024-08-21 22:45:20 +0800215 std::cerr << "Error opening " << sysPath << "\n";
216 return 0;
James Feist6714a252018-09-10 15:26:18 -0700217 }
218 std::string line;
219 if (!std::getline(ref, line))
220 {
Yang Chene1dcb5c2024-08-21 22:45:20 +0800221 std::cerr << "Error reading pwm at " << sysPath << "\n";
222 return 0;
James Feist6714a252018-09-10 15:26:18 -0700223 }
224 try
225 {
226 uint32_t value = std::stoi(line);
227 return value;
228 }
Patrick Williams26601e82021-10-06 12:43:25 -0500229 catch (const std::invalid_argument&)
James Feist6714a252018-09-10 15:26:18 -0700230 {
Yang Chene1dcb5c2024-08-21 22:45:20 +0800231 std::cerr << "Error converting pwm\n";
James Feist6714a252018-09-10 15:26:18 -0700232 // throw if not initial read to be caught by dbus bindings
233 if (errThrow)
234 {
235 throw std::runtime_error("Bad Read");
236 }
237 }
238 return 0;
239}