1e10416ecSMatt Spinler /** 2e10416ecSMatt Spinler * Copyright © 2017 IBM Corporation 3e10416ecSMatt Spinler * 4e10416ecSMatt Spinler * Licensed under the Apache License, Version 2.0 (the "License"); 5e10416ecSMatt Spinler * you may not use this file except in compliance with the License. 6e10416ecSMatt Spinler * You may obtain a copy of the License at 7e10416ecSMatt Spinler * 8e10416ecSMatt Spinler * http://www.apache.org/licenses/LICENSE-2.0 9e10416ecSMatt Spinler * 10e10416ecSMatt Spinler * Unless required by applicable law or agreed to in writing, software 11e10416ecSMatt Spinler * distributed under the License is distributed on an "AS IS" BASIS, 12e10416ecSMatt Spinler * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13e10416ecSMatt Spinler * See the License for the specific language governing permissions and 14e10416ecSMatt Spinler * limitations under the License. 15e10416ecSMatt Spinler */ 16*3e1bb274SMatthew Barth #include "config.h" 17*3e1bb274SMatthew Barth 18*3e1bb274SMatthew Barth #include "manager.hpp" 19*3e1bb274SMatthew Barth 20*3e1bb274SMatthew Barth #include "sdbusplus.hpp" 21*3e1bb274SMatthew Barth #include "utility.hpp" 22*3e1bb274SMatthew Barth 23*3e1bb274SMatthew Barth #include <unistd.h> 24*3e1bb274SMatthew Barth 25*3e1bb274SMatthew Barth #include <phosphor-logging/elog-errors.hpp> 26*3e1bb274SMatthew Barth #include <phosphor-logging/elog.hpp> 27*3e1bb274SMatthew Barth #include <phosphor-logging/log.hpp> 28*3e1bb274SMatthew Barth #include <sdbusplus/bus.hpp> 29*3e1bb274SMatthew Barth #include <xyz/openbmc_project/Common/error.hpp> 30*3e1bb274SMatthew Barth 3157352a31SMatt Spinler #include <algorithm> 3293af4194SMatthew Barth #include <experimental/filesystem> 33e10416ecSMatt Spinler 34e10416ecSMatt Spinler namespace phosphor 35e10416ecSMatt Spinler { 36e10416ecSMatt Spinler namespace fan 37e10416ecSMatt Spinler { 38e10416ecSMatt Spinler namespace control 39e10416ecSMatt Spinler { 40e10416ecSMatt Spinler 41ee7f6428SMatt Spinler using namespace phosphor::logging; 4293af4194SMatthew Barth namespace fs = std::experimental::filesystem; 43ee7f6428SMatt Spinler 44ee7f6428SMatt Spinler constexpr auto SYSTEMD_SERVICE = "org.freedesktop.systemd1"; 45ee7f6428SMatt Spinler constexpr auto SYSTEMD_OBJ_PATH = "/org/freedesktop/systemd1"; 46ee7f6428SMatt Spinler constexpr auto SYSTEMD_INTERFACE = "org.freedesktop.systemd1.Manager"; 47ee7f6428SMatt Spinler constexpr auto FAN_CONTROL_READY_TARGET = "obmc-fan-control-ready@0.target"; 48ee7f6428SMatt Spinler 49f96b01e2SGunnar Mills /** 50f96b01e2SGunnar Mills * Check if a condition is true. Conditions are used to determine 51f96b01e2SGunnar Mills * which fan zone to use. 52f96b01e2SGunnar Mills * 53f96b01e2SGunnar Mills * @param[in] bus - The D-Bus bus object 54f96b01e2SGunnar Mills * @param[in] condition - The condition to check if true 55f96b01e2SGunnar Mills * @return result - True if the condition is true 56f96b01e2SGunnar Mills */ 57f0b020fbSBrad Bishop bool checkCondition(sdbusplus::bus::bus& bus, const Condition& c) 58f96b01e2SGunnar Mills { 59f96b01e2SGunnar Mills auto& type = std::get<conditionTypePos>(c); 60f96b01e2SGunnar Mills auto& properties = std::get<conditionPropertyListPos>(c); 61f96b01e2SGunnar Mills 62f96b01e2SGunnar Mills for (auto& p : properties) 63f96b01e2SGunnar Mills { 64803d35bcSMatthew Barth auto value = std::get<propertyValuePos>(p); 65f96b01e2SGunnar Mills 66f96b01e2SGunnar Mills // TODO openbmc/openbmc#1769: Support more types than just getProperty. 67f96b01e2SGunnar Mills if (type.compare("getProperty") == 0) 68f96b01e2SGunnar Mills { 69803d35bcSMatthew Barth auto propertyValue = util::SDBusPlus::getProperty<decltype(value)>( 70*3e1bb274SMatthew Barth bus, std::get<propertyPathPos>(p), 71f96b01e2SGunnar Mills std::get<propertyInterfacePos>(p), 72803d35bcSMatthew Barth std::get<propertyNamePos>(p)); 73f96b01e2SGunnar Mills 74f96b01e2SGunnar Mills if (value != propertyValue) 75f96b01e2SGunnar Mills { 76f96b01e2SGunnar Mills return false; 77f96b01e2SGunnar Mills } 78f96b01e2SGunnar Mills } 79f96b01e2SGunnar Mills } 80f96b01e2SGunnar Mills return true; 81f96b01e2SGunnar Mills } 82f96b01e2SGunnar Mills 83ee7f6428SMatt Spinler // Note: Future code will check 'mode' before starting control algorithm 84*3e1bb274SMatthew Barth Manager::Manager(sdbusplus::bus::bus& bus, const sdeventplus::Event& event, 85ee7f6428SMatt Spinler Mode mode) : 8693af4194SMatthew Barth _bus(bus), 8793af4194SMatthew Barth _objMgr(bus, CONTROL_OBJPATH) 88e10416ecSMatt Spinler { 8957352a31SMatt Spinler // Create the appropriate Zone objects based on the 9057352a31SMatt Spinler // actual system configuration. 9157352a31SMatt Spinler 9257352a31SMatt Spinler // Find the 1 ZoneGroup that meets all of its conditions 9357352a31SMatt Spinler for (auto& group : _zoneLayouts) 9457352a31SMatt Spinler { 9557352a31SMatt Spinler auto& conditions = std::get<conditionListPos>(group); 9657352a31SMatt Spinler 9757352a31SMatt Spinler if (std::all_of(conditions.begin(), conditions.end(), 98*3e1bb274SMatthew Barth [&bus](const auto& condition) { 99f96b01e2SGunnar Mills return checkCondition(bus, condition); 10057352a31SMatt Spinler })) 10157352a31SMatt Spinler { 10257352a31SMatt Spinler // Create a Zone object for each zone in this group 10357352a31SMatt Spinler auto& zones = std::get<zoneListPos>(group); 10457352a31SMatt Spinler 10557352a31SMatt Spinler for (auto& z : zones) 10657352a31SMatt Spinler { 10793af4194SMatthew Barth fs::path path{CONTROL_OBJPATH}; 10893af4194SMatthew Barth path /= std::to_string(std::get<zoneNumPos>(z)); 10957352a31SMatt Spinler _zones.emplace(std::get<zoneNumPos>(z), 110*3e1bb274SMatthew Barth std::make_unique<Zone>(mode, _bus, path.string(), 111*3e1bb274SMatthew Barth event, z)); 11257352a31SMatt Spinler } 11357352a31SMatt Spinler 11457352a31SMatt Spinler break; 11557352a31SMatt Spinler } 11657352a31SMatt Spinler } 11757352a31SMatt Spinler 11893af4194SMatthew Barth if (mode == Mode::control) 11993af4194SMatthew Barth { 12014cc043fSMatthew Barth bus.request_name(CONTROL_BUSNAME); 121ee7f6428SMatt Spinler } 12293af4194SMatthew Barth } 123ee7f6428SMatt Spinler 124ee7f6428SMatt Spinler void Manager::doInit() 125ee7f6428SMatt Spinler { 12657352a31SMatt Spinler for (auto& z : _zones) 12757352a31SMatt Spinler { 12857352a31SMatt Spinler z.second->setFullSpeed(); 12957352a31SMatt Spinler } 130ee7f6428SMatt Spinler 131ee7f6428SMatt Spinler auto delay = _powerOnDelay; 132ee7f6428SMatt Spinler while (delay > 0) 133ee7f6428SMatt Spinler { 134ee7f6428SMatt Spinler delay = sleep(delay); 135e10416ecSMatt Spinler } 136e10416ecSMatt Spinler 137*3e1bb274SMatthew Barth util::SDBusPlus::callMethod(_bus, SYSTEMD_SERVICE, SYSTEMD_OBJ_PATH, 138*3e1bb274SMatthew Barth SYSTEMD_INTERFACE, "StartUnit", 139*3e1bb274SMatthew Barth FAN_CONTROL_READY_TARGET, "replace"); 140ee7f6428SMatt Spinler } 141e10416ecSMatt Spinler 142f96b01e2SGunnar Mills } // namespace control 143f96b01e2SGunnar Mills } // namespace fan 144f96b01e2SGunnar Mills } // namespace phosphor 145