blob: 849cd918d33ea73a3d45014c2edbf31310cac11c [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(
Matthew Barth34f1bda2017-05-31 13:45:36 -050073 std::make_unique<EventData>(
Matthew Barth14184132017-05-19 14:37:30 -050074 EventData
75 {
76 std::get<groupPos>(event),
77 std::get<handlerObjPos>(prop),
78 std::get<actionPos>(event)
79 }));
80 _matches.emplace_back(
81 bus,
82 std::get<signaturePos>(prop).c_str(),
Matthew Barth34f1bda2017-05-31 13:45:36 -050083 std::bind(std::mem_fn(&Zone::handleEvent),
84 this,
85 std::placeholders::_1,
86 _signalEvents.back().get()));
Matthew Barth14184132017-05-19 14:37:30 -050087 }
Matthew Barthdf3e8d62017-05-31 11:07:24 -050088 // Run action function for initial event state
89 std::get<actionPos>(event)(*this,
90 std::get<groupPos>(event));
Matthew Barth17d1fe22017-05-11 15:00:36 -050091 }
Matthew Barth38a93a82017-05-11 14:12:27 -050092 }
Matt Spinler7f88fe62017-04-10 14:39:02 -050093}
94
95
96void Zone::setSpeed(uint64_t speed)
97{
98 for (auto& fan : _fans)
99 {
100 fan->setSpeed(speed);
101 }
102}
103
Matthew Barth861d77c2017-05-22 14:18:25 -0500104void Zone::setActiveAllow(const Group* group, bool isActiveAllow)
105{
106 _active[group] = isActiveAllow;
107 if (!isActiveAllow)
108 {
109 _isActive = false;
110 }
111 else
112 {
113 // Check all entries are set to allow control active
114 auto actPred = [](auto const& entry) {return entry.second;};
115 _isActive = std::all_of(_active.begin(),
116 _active.end(),
117 actPred);
118 }
119}
120
Matthew Barthdf3e8d62017-05-31 11:07:24 -0500121template <typename T>
122void Zone::getProperty(sdbusplus::bus::bus& bus,
123 const std::string& path,
124 const std::string& iface,
125 const std::string& prop,
126 T& value)
127{
128 sdbusplus::message::variant<T> property;
129 auto serv = phosphor::fan::util::getService(path, iface, bus);
130 auto hostCall = bus.new_method_call(serv.c_str(),
131 path.c_str(),
132 "org.freedesktop.DBus.Properties",
133 "Get");
134 hostCall.append(iface);
135 hostCall.append(prop);
136 auto hostResponseMsg = bus.call(hostCall);
137 if (hostResponseMsg.is_method_error())
138 {
139 throw std::runtime_error(
140 "Error in host call response for retrieving property");
141 }
142 hostResponseMsg.read(property);
143 value = sdbusplus::message::variant_ns::get<T>(property);
144}
145
Matthew Barth38a93a82017-05-11 14:12:27 -0500146void Zone::handleEvent(sdbusplus::message::message& msg,
Matthew Barth34f1bda2017-05-31 13:45:36 -0500147 const EventData* eventData)
Matthew Barth38a93a82017-05-11 14:12:27 -0500148{
149 // Handle the callback
Matthew Barth34f1bda2017-05-31 13:45:36 -0500150 std::get<eventHandlerPos>(*eventData)(_bus, msg, *this);
Matthew Barth17d1fe22017-05-11 15:00:36 -0500151 // Perform the action
Matthew Barth34f1bda2017-05-31 13:45:36 -0500152 std::get<eventActionPos>(*eventData)(*this,
153 std::get<eventGroupPos>(*eventData));
Matthew Barth38a93a82017-05-11 14:12:27 -0500154}
155
Matt Spinler7f88fe62017-04-10 14:39:02 -0500156}
157}
158}