/** * Copyright © 2017 IBM Corporation * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include "config.h" #include "manager.hpp" #include "sdbusplus.hpp" #include "utility.hpp" #include #include #include #include #include #include #include #include namespace phosphor { namespace fan { namespace control { using namespace phosphor::logging; namespace fs = std::experimental::filesystem; constexpr auto SYSTEMD_SERVICE = "org.freedesktop.systemd1"; constexpr auto SYSTEMD_OBJ_PATH = "/org/freedesktop/systemd1"; constexpr auto SYSTEMD_INTERFACE = "org.freedesktop.systemd1.Manager"; constexpr auto FAN_CONTROL_READY_TARGET = "obmc-fan-control-ready@0.target"; /** * Check if a condition is true. Conditions are used to determine * which fan zone to use. * * @param[in] bus - The D-Bus bus object * @param[in] condition - The condition to check if true * @return result - True if the condition is true */ bool checkCondition(sdbusplus::bus::bus& bus, const Condition& c) { auto& type = std::get(c); auto& properties = std::get(c); for (auto& p : properties) { auto value = std::get(p); // TODO openbmc/openbmc#1769: Support more types than just getProperty. if (type.compare("getProperty") == 0) { auto propertyValue = util::SDBusPlus::getProperty( bus, std::get(p), std::get(p), std::get(p)); if (value != propertyValue) { return false; } } } return true; } // Note: Future code will check 'mode' before starting control algorithm Manager::Manager(sdbusplus::bus::bus& bus, const sdeventplus::Event& event, Mode mode) : _bus(bus), _objMgr(bus, CONTROL_OBJPATH) { // Create the appropriate Zone objects based on the // actual system configuration. // Find the 1 ZoneGroup that meets all of its conditions for (auto& group : _zoneLayouts) { auto& conditions = std::get(group); if (std::all_of(conditions.begin(), conditions.end(), [&bus](const auto& condition) { return checkCondition(bus, condition); })) { // Create a Zone object for each zone in this group auto& zones = std::get(group); for (auto& z : zones) { fs::path path{CONTROL_OBJPATH}; path /= std::to_string(std::get(z)); _zones.emplace(std::get(z), std::make_unique(mode, _bus, path.string(), event, z)); } break; } } if (mode == Mode::control) { bus.request_name(CONTROL_BUSNAME); } } void Manager::doInit() { for (auto& z : _zones) { z.second->setFullSpeed(); } auto delay = _powerOnDelay; while (delay > 0) { delay = sleep(delay); } util::SDBusPlus::callMethod(_bus, SYSTEMD_SERVICE, SYSTEMD_OBJ_PATH, SYSTEMD_INTERFACE, "StartUnit", FAN_CONTROL_READY_TARGET, "replace"); } } // namespace control } // namespace fan } // namespace phosphor