blob: dff11d6355e29456c1e88e50193665456944036b [file] [log] [blame]
James Feist7136a5a2018-07-19 09:52:05 -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*/
16
Patrick Ventureda4a5dd2018-08-31 09:42:48 -070017#include "dbus/dbuswrite.hpp"
18
James Feist7136a5a2018-07-19 09:52:05 -070019#include <iostream>
Patrick Venturef5e770b2018-10-30 12:28:53 -070020#include <memory>
Patrick Venture76ce5d72018-10-30 18:39:50 -070021#include <phosphor-logging/log.hpp>
James Feist7136a5a2018-07-19 09:52:05 -070022#include <sdbusplus/bus.hpp>
Patrick Venturef5e770b2018-10-30 12:28:53 -070023#include <string>
24
25constexpr const char* pwmInterface = "xyz.openbmc_project.Control.FanPwm";
James Feist7136a5a2018-07-19 09:52:05 -070026
Patrick Venture76ce5d72018-10-30 18:39:50 -070027using namespace phosphor::logging;
28
James Feist7136a5a2018-07-19 09:52:05 -070029// this bus object is treated as a singleton because the class is constructed in
30// a different thread than it is used, and as bus objects are relatively
31// expensive we'd prefer to only have one
32std::unique_ptr<sdbusplus::bus::bus> writeBus = nullptr;
33
Patrick Venturef5e770b2018-10-30 12:28:53 -070034std::unique_ptr<WriteInterface>
35 DbusWritePercent::createDbusWrite(const std::string& path, int64_t min,
36 int64_t max, DbusHelperInterface& helper)
37{
38 auto tempBus = sdbusplus::bus::new_default();
39 std::string connectionName;
40
41 try
42 {
43 connectionName = helper.getService(tempBus, pwmInterface, path);
44 }
45 catch (const std::exception& e)
46 {
47 return nullptr;
48 }
49
50 return std::make_unique<DbusWritePercent>(path, min, max, connectionName);
51}
52
James Feist7136a5a2018-07-19 09:52:05 -070053void initBus()
54{
55 if (writeBus == nullptr)
56 {
57 writeBus = std::make_unique<sdbusplus::bus::bus>(
58 sdbusplus::bus::new_default());
59 }
60}
61
62void DbusWritePercent::write(double value)
63{
Patrick Venture5f59c0f2018-11-11 12:55:14 -080064 double minimum = getMin();
65 double maximum = getMax();
James Feist7136a5a2018-07-19 09:52:05 -070066
Patrick Venture5f59c0f2018-11-11 12:55:14 -080067 double range = maximum - minimum;
68 double offset = range * value;
69 double ovalue = offset + minimum;
James Feistcd9e1092018-10-08 13:06:41 -070070
71 if (oldValue == static_cast<int64_t>(ovalue))
72 {
73 return;
74 }
James Feist7136a5a2018-07-19 09:52:05 -070075 initBus();
76 auto mesg =
77 writeBus->new_method_call(connectionName.c_str(), path.c_str(),
78 "org.freedesktop.DBus.Properties", "Set");
79 mesg.append(pwmInterface, "Target",
80 sdbusplus::message::variant<uint64_t>(ovalue));
Patrick Venture76ce5d72018-10-30 18:39:50 -070081
82 try
James Feist7136a5a2018-07-19 09:52:05 -070083 {
Patrick Venture76ce5d72018-10-30 18:39:50 -070084 // TODO: if we don't use the reply, call_noreply()
85 auto resp = writeBus->call(mesg);
James Feist7136a5a2018-07-19 09:52:05 -070086 }
Patrick Venture76ce5d72018-10-30 18:39:50 -070087 catch (const sdbusplus::exception::SdBusError& ex)
88 {
89 log<level::ERR>("Dbus Call Failure", entry("PATH=%s", path.c_str()),
90 entry("WHAT=%s", ex.what()));
91 }
92
James Feistcd9e1092018-10-08 13:06:41 -070093 oldValue = static_cast<int64_t>(ovalue);
James Feist7136a5a2018-07-19 09:52:05 -070094 return;
95}
96
Patrick Venturef5e770b2018-10-30 12:28:53 -070097std::unique_ptr<WriteInterface>
98 DbusWrite::createDbusWrite(const std::string& path, int64_t min,
99 int64_t max, DbusHelperInterface& helper)
100{
101 auto tempBus = sdbusplus::bus::new_default();
102 std::string connectionName;
103
104 try
105 {
106 connectionName = helper.getService(tempBus, pwmInterface, path);
107 }
108 catch (const std::exception& e)
109 {
110 return nullptr;
111 }
112
113 return std::make_unique<DbusWrite>(path, min, max, connectionName);
114}
115
James Feist7136a5a2018-07-19 09:52:05 -0700116void DbusWrite::write(double value)
117{
James Feistcd9e1092018-10-08 13:06:41 -0700118 if (oldValue == static_cast<int64_t>(value))
119 {
120 return;
121 }
James Feist7136a5a2018-07-19 09:52:05 -0700122 initBus();
123 auto mesg =
124 writeBus->new_method_call(connectionName.c_str(), path.c_str(),
125 "org.freedesktop.DBus.Properties", "Set");
126 mesg.append(pwmInterface, "Target",
127 sdbusplus::message::variant<uint64_t>(value));
Patrick Venture76ce5d72018-10-30 18:39:50 -0700128
129 try
James Feist7136a5a2018-07-19 09:52:05 -0700130 {
Patrick Venture76ce5d72018-10-30 18:39:50 -0700131 // TODO: consider call_noreplly
132 auto resp = writeBus->call(mesg);
James Feist7136a5a2018-07-19 09:52:05 -0700133 }
Patrick Venture76ce5d72018-10-30 18:39:50 -0700134 catch (const sdbusplus::exception::SdBusError& ex)
135 {
136 log<level::ERR>("Dbus Call Failure", entry("PATH=%s", path.c_str()),
137 entry("WHAT=%s", ex.what()));
138 }
139
James Feistcd9e1092018-10-08 13:06:41 -0700140 oldValue = static_cast<int64_t>(value);
James Feist7136a5a2018-07-19 09:52:05 -0700141 return;
142}