blob: de86170bb768bff889066beb9046c2d1d873d8e8 [file] [log] [blame]
Matthew Barthbf7b7b12017-03-07 15:46:59 -06001#pragma once
2
3#include <experimental/filesystem>
Matt Spinlerb4e65572017-10-17 10:09:18 -05004#include <phosphor-logging/elog-errors.hpp>
5#include <phosphor-logging/log.hpp>
6#include <xyz/openbmc_project/Sensor/Device/error.hpp>
Matthew Barth048ac872017-03-09 14:36:08 -06007#include "fan_speed.hpp"
Patrick Venture9331ab72018-01-29 09:48:47 -08008#include "fan_pwm.hpp"
Matthew Barthbf7b7b12017-03-07 15:46:59 -06009
10/** @class Targets
11 * @brief Target type traits.
12 *
13 * @tparam T - The target type.
14 */
15template <typename T>
16struct Targets
17{
18 static void fail()
19 {
20 static_assert(sizeof(Targets) == -1, "Unsupported Target type");
21 }
22};
23
24/**@brief Targets specialization for fan speed. */
25template <>
Matthew Barth048ac872017-03-09 14:36:08 -060026struct Targets<hwmon::FanSpeed>
Matthew Barthbf7b7b12017-03-07 15:46:59 -060027{
28 static constexpr InterfaceType type = InterfaceType::FAN_SPEED;
Matthew Barthbf7b7b12017-03-07 15:46:59 -060029};
30
Patrick Venture9331ab72018-01-29 09:48:47 -080031template <>
32struct Targets<hwmon::FanPwm>
33{
34 static constexpr InterfaceType type = InterfaceType::FAN_PWM;
35};
36
Matthew Barthbf7b7b12017-03-07 15:46:59 -060037/** @brief addTarget
38 *
39 * Creates the target type interface
40 *
41 * @tparam T - The target type
42 *
43 * @param[in] sensor - A sensor type and name
Matt Spinlerb4e65572017-10-17 10:09:18 -050044 * @param[in] ioAccess - hwmon sysfs access object
45 * @param[in] devPath - The /sys/devices sysfs path
Matthew Barthbf7b7b12017-03-07 15:46:59 -060046 * @param[in] info - The sdbusplus server connection and interfaces
Matt Spinler0a8de642017-05-11 10:59:39 -050047 *
48 * @return A shared pointer to the target interface object
49 * Will be empty if no interface was created
Matthew Barthbf7b7b12017-03-07 15:46:59 -060050 */
51template <typename T>
Matt Spinler0a8de642017-05-11 10:59:39 -050052std::shared_ptr<T> addTarget(const SensorSet::key_type& sensor,
Matt Spinlerb4e65572017-10-17 10:09:18 -050053 const sysfs::hwmonio::HwmonIO& ioAccess,
Brad Bishop751043e2017-08-29 11:13:46 -040054 const std::string& devPath,
Matt Spinler0a8de642017-05-11 10:59:39 -050055 ObjectInfo& info)
Matthew Barthbf7b7b12017-03-07 15:46:59 -060056{
Matt Spinler0a8de642017-05-11 10:59:39 -050057 std::shared_ptr<T> target;
Matthew Barthbf7b7b12017-03-07 15:46:59 -060058 namespace fs = std::experimental::filesystem;
59 static constexpr bool deferSignals = true;
60
61 auto& bus = *std::get<sdbusplus::bus::bus*>(info);
62 auto& obj = std::get<Object>(info);
63 auto& objPath = std::get<std::string>(info);
Patrick Venture9331ab72018-01-29 09:48:47 -080064 auto type = Targets<T>::type;
Matthew Barthbf7b7b12017-03-07 15:46:59 -060065
66 // Check if target sysfs file exists
Patrick Venture9331ab72018-01-29 09:48:47 -080067 std::string sysfsFullPath;
Lei YUa4bba962018-01-31 13:51:05 +080068 std::string targetName = sensor.first;
69 std::string targetId = sensor.second;
70 std::string entry = hwmon::entry::target;
Patrick Venture9331ab72018-01-29 09:48:47 -080071
72 using namespace std::literals;
73 const std::string pwm = "pwm"s;
74 const std::string empty = ""s;
75
76 if (InterfaceType::FAN_PWM == type)
77 {
Lei YUa4bba962018-01-31 13:51:05 +080078 targetName = pwm;
79 // If PWM_TARGET is set, use the specified pwm id
80 auto id = getEnv("PWM_TARGET", sensor);
81 if (!id.empty())
82 {
83 targetId = id;
84 }
85 entry = empty;
Patrick Venture9331ab72018-01-29 09:48:47 -080086 }
87
Lei YUa4bba962018-01-31 13:51:05 +080088 sysfsFullPath = sysfs::make_sysfs_path(ioAccess.path(),
89 targetName,
90 targetId,
91 entry);
Matthew Barth048ac872017-03-09 14:36:08 -060092 if (fs::exists(sysfsFullPath))
Matthew Barthbf7b7b12017-03-07 15:46:59 -060093 {
Matt Spinlerb4e65572017-10-17 10:09:18 -050094 uint32_t targetSpeed = 0;
95
96 try
97 {
Lei YUa4bba962018-01-31 13:51:05 +080098 targetSpeed = ioAccess.read(
99 targetName,
100 targetId,
101 entry,
102 sysfs::hwmonio::retries,
103 sysfs::hwmonio::delay);
Matt Spinlerb4e65572017-10-17 10:09:18 -0500104 }
105 catch (const std::system_error& e)
106 {
107 using namespace phosphor::logging;
108 using namespace sdbusplus::xyz::openbmc_project::
109 Sensor::Device::Error;
110 using metadata = xyz::openbmc_project::Sensor::
111 Device::ReadFailure;
112
113 report<ReadFailure>(
114 metadata::CALLOUT_ERRNO(e.code().value()),
115 metadata::CALLOUT_DEVICE_PATH(devPath.c_str()));
116
117 log<level::INFO>("Logging failing sysfs file",
118 phosphor::logging::entry(
119 "FILE=%s", sysfsFullPath.c_str()));
120 }
121
122 target = std::make_shared<T>(ioAccess.path(),
Brad Bishop751043e2017-08-29 11:13:46 -0400123 devPath,
Lei YUa4bba962018-01-31 13:51:05 +0800124 targetId,
Matt Spinler0a8de642017-05-11 10:59:39 -0500125 bus,
126 objPath.c_str(),
Matt Spinlerb4e65572017-10-17 10:09:18 -0500127 deferSignals,
128 targetSpeed);
Matt Spinler0a8de642017-05-11 10:59:39 -0500129 obj[type] = target;
Matthew Barthbf7b7b12017-03-07 15:46:59 -0600130 }
Matt Spinler0a8de642017-05-11 10:59:39 -0500131
132 return target;
Matthew Barthbf7b7b12017-03-07 15:46:59 -0600133}