blob: 13ee977f8e4a4fd4ce0c3085a78645abd6d76321 [file] [log] [blame]
Matt Spinler7f88fe62017-04-10 14:39:02 -05001/**
2 * Copyright © 2017 IBM 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 */
Matthew Barthdf3e8d62017-05-31 11:07:24 -050016#include <phosphor-logging/elog.hpp>
Matt Spinler7f88fe62017-04-10 14:39:02 -050017#include "zone.hpp"
Matthew Barthdf3e8d62017-05-31 11:07:24 -050018#include "utility.hpp"
Matt Spinler7f88fe62017-04-10 14:39:02 -050019
20namespace phosphor
21{
22namespace fan
23{
24namespace control
25{
26
Matthew Barthdf3e8d62017-05-31 11:07:24 -050027using namespace phosphor::logging;
Matt Spinler7f88fe62017-04-10 14:39:02 -050028
Matthew Barth14184132017-05-19 14:37:30 -050029Zone::Zone(Mode mode,
30 sdbusplus::bus::bus& bus,
Matt Spinler7f88fe62017-04-10 14:39:02 -050031 const ZoneDefinition& def) :
32 _bus(bus),
33 _fullSpeed(std::get<fullSpeedPos>(def)),
34 _zoneNum(std::get<zoneNumPos>(def))
35{
36 auto& fanDefs = std::get<fanListPos>(def);
37
38 for (auto& def : fanDefs)
39 {
40 _fans.emplace_back(std::make_unique<Fan>(bus, def));
41 }
Matthew Barth38a93a82017-05-11 14:12:27 -050042
Matthew Barth14184132017-05-19 14:37:30 -050043 // Do not enable set speed events when in init mode
44 if (mode != Mode::init)
Matthew Barth17d1fe22017-05-11 15:00:36 -050045 {
Matthew Barth14184132017-05-19 14:37:30 -050046 // Setup signal trigger for set speed events
47 for (auto& event : std::get<setSpeedEventsPos>(def))
Matthew Barth17d1fe22017-05-11 15:00:36 -050048 {
Matthew Barthdf3e8d62017-05-31 11:07:24 -050049 // Get the current value for each property
50 for (auto& entry : std::get<groupPos>(event))
51 {
52 try
53 {
54 bool value = false;
55 getProperty(_bus,
56 entry.first,
57 std::get<intfPos>(entry.second),
58 std::get<propPos>(entry.second),
59 value);
60 setPropertyValue(entry.first.c_str(),
61 std::get<propPos>(entry.second).c_str(),
62 value);
63 }
64 catch (const std::exception& e)
65 {
66 log<level::ERR>(e.what());
67 }
68 }
69 // Setup signal matches for property change events
Matthew Barth14184132017-05-19 14:37:30 -050070 for (auto& prop : std::get<propChangeListPos>(event))
71 {
72 _signalEvents.emplace_back(
73 std::make_unique<SignalEvent>(
74 this,
75 EventData
76 {
77 std::get<groupPos>(event),
78 std::get<handlerObjPos>(prop),
79 std::get<actionPos>(event)
80 }));
81 _matches.emplace_back(
82 bus,
83 std::get<signaturePos>(prop).c_str(),
84 signalHandler,
85 _signalEvents.back().get());
86 }
Matthew Barthdf3e8d62017-05-31 11:07:24 -050087 // Run action function for initial event state
88 std::get<actionPos>(event)(*this,
89 std::get<groupPos>(event));
Matthew Barth17d1fe22017-05-11 15:00:36 -050090 }
Matthew Barth38a93a82017-05-11 14:12:27 -050091 }
Matt Spinler7f88fe62017-04-10 14:39:02 -050092}
93
94
95void Zone::setSpeed(uint64_t speed)
96{
97 for (auto& fan : _fans)
98 {
99 fan->setSpeed(speed);
100 }
101}
102
Matthew Barth861d77c2017-05-22 14:18:25 -0500103void Zone::setActiveAllow(const Group* group, bool isActiveAllow)
104{
105 _active[group] = isActiveAllow;
106 if (!isActiveAllow)
107 {
108 _isActive = false;
109 }
110 else
111 {
112 // Check all entries are set to allow control active
113 auto actPred = [](auto const& entry) {return entry.second;};
114 _isActive = std::all_of(_active.begin(),
115 _active.end(),
116 actPred);
117 }
118}
119
Matthew Barthdf3e8d62017-05-31 11:07:24 -0500120template <typename T>
121void Zone::getProperty(sdbusplus::bus::bus& bus,
122 const std::string& path,
123 const std::string& iface,
124 const std::string& prop,
125 T& value)
126{
127 sdbusplus::message::variant<T> property;
128 auto serv = phosphor::fan::util::getService(path, iface, bus);
129 auto hostCall = bus.new_method_call(serv.c_str(),
130 path.c_str(),
131 "org.freedesktop.DBus.Properties",
132 "Get");
133 hostCall.append(iface);
134 hostCall.append(prop);
135 auto hostResponseMsg = bus.call(hostCall);
136 if (hostResponseMsg.is_method_error())
137 {
138 throw std::runtime_error(
139 "Error in host call response for retrieving property");
140 }
141 hostResponseMsg.read(property);
142 value = sdbusplus::message::variant_ns::get<T>(property);
143}
144
Matthew Barth38a93a82017-05-11 14:12:27 -0500145int Zone::signalHandler(sd_bus_message* msg,
146 void* data,
147 sd_bus_error* err)
148{
149 auto sdbpMsg = sdbusplus::message::message(msg);
150 auto& signalEvent = *static_cast<SignalEvent*>(data);
151 std::get<zoneObjPos>(signalEvent)->handleEvent(
152 sdbpMsg,
Matthew Barth17d1fe22017-05-11 15:00:36 -0500153 std::get<eventDataPos>(signalEvent));
Matthew Barth38a93a82017-05-11 14:12:27 -0500154 return 0;
155}
156
157void Zone::handleEvent(sdbusplus::message::message& msg,
Matthew Barth17d1fe22017-05-11 15:00:36 -0500158 const EventData& eventData)
Matthew Barth38a93a82017-05-11 14:12:27 -0500159{
160 // Handle the callback
Matthew Barth17d1fe22017-05-11 15:00:36 -0500161 std::get<eventHandlerPos>(eventData)(_bus, msg, *this);
162 // Perform the action
163 std::get<eventActionPos>(eventData)(*this,
164 std::get<eventGroupPos>(eventData));
Matthew Barth38a93a82017-05-11 14:12:27 -0500165}
166
Matt Spinler7f88fe62017-04-10 14:39:02 -0500167}
168}
169}