blob: 3ba365cb33d1d15c02ee951e477834a6af111bf6 [file] [log] [blame]
Brad Bishope55ef3d2016-12-19 09:12:40 -05001/**
2 * Copyright © 2016 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 */
16#include <iostream>
17#include <memory>
Brad Bishop9c7b6e02016-12-19 12:43:36 -050018#include <cstring>
19#include <cstdlib>
20#include <chrono>
Brad Bishope55ef3d2016-12-19 09:12:40 -050021#include "sensorset.hpp"
22#include "sensorcache.hpp"
23#include "hwmon.hpp"
24#include "sysfs.hpp"
Brad Bishopd499ca62016-12-19 09:24:50 -050025#include "mainloop.hpp"
Brad Bishope55ef3d2016-12-19 09:12:40 -050026
Brad Bishop9c7b6e02016-12-19 12:43:36 -050027using namespace std::literals::chrono_literals;
28
Brad Bishopb9e2b072016-12-19 13:47:10 -050029MainLoop::MainLoop(
Brad Bishop9c7b6e02016-12-19 12:43:36 -050030 sdbusplus::bus::bus&& bus,
Brad Bishopb9e2b072016-12-19 13:47:10 -050031 const std::string& path,
32 const char* prefix,
33 const char* root)
Brad Bishop9c7b6e02016-12-19 12:43:36 -050034 : _bus(std::move(bus)),
35 _manager(sdbusplus::server::manager::manager(_bus, root)),
36 _shutdown(false),
Brad Bishopb9e2b072016-12-19 13:47:10 -050037 _path(path),
38 _prefix(prefix),
Brad Bishop3c344d32017-01-05 11:48:39 -050039 _root(root),
40 state()
Brad Bishopd499ca62016-12-19 09:24:50 -050041{
Brad Bishop9c7b6e02016-12-19 12:43:36 -050042 if (_path.back() == '/')
43 {
44 _path.pop_back();
45 }
Brad Bishopd499ca62016-12-19 09:24:50 -050046}
47
48void MainLoop::shutdown() noexcept
49{
50 _shutdown = true;
51}
52
53void MainLoop::run()
Brad Bishope55ef3d2016-12-19 09:12:40 -050054{
55 // Check sysfs for available sensors.
Brad Bishopd499ca62016-12-19 09:24:50 -050056 auto sensors = std::make_unique<SensorSet>(_path);
Brad Bishope55ef3d2016-12-19 09:12:40 -050057 auto sensor_cache = std::make_unique<SensorCache>();
58
Brad Bishop75b4ab82017-01-06 09:33:50 -050059 for (auto& i : *sensors)
60 {
Brad Bishop73831cd2017-01-06 09:37:22 -050061 // Ignore inputs without a label.
62 std::string envKey = "LABEL_" + i.first.first + i.first.second;
63 std::string label;
64
65 auto env = getenv(envKey.c_str());
66
67 if (env)
68 {
69 label.assign(env);
70 }
71
72 if (label.empty())
73 {
74 continue;
75 }
76
Brad Bishop075f7a22017-01-06 09:45:08 -050077 Object o;
78 std::string objectPath{_root};
79
80 objectPath.append("/");
81 objectPath.append(i.first.first);
82 objectPath.append("/");
83 objectPath.append(label);
84
85 auto iface = std::make_shared<ValueObject>(_bus, objectPath.c_str());
86 o.emplace(InterfaceType::VALUE, iface);
87
88 auto value = std::make_tuple(
89 std::move(i.second),
90 std::move(label),
91 std::move(o));
Brad Bishop73831cd2017-01-06 09:37:22 -050092
Brad Bishop75b4ab82017-01-06 09:33:50 -050093 state[std::move(i.first)] = std::move(value);
94 }
95
Brad Bishop9c7b6e02016-12-19 12:43:36 -050096 {
97 struct Free
98 {
99 void operator()(char* ptr) const
100 {
101 free(ptr);
102 }
103 };
104
105 auto copy = std::unique_ptr<char, Free>(strdup(_path.c_str()));
106 auto busname = std::string(_prefix) + '.' + basename(copy.get());
107 _bus.request_name(busname.c_str());
108 }
109
Brad Bishope55ef3d2016-12-19 09:12:40 -0500110 // TODO: Issue#3 - Need to make calls to the dbus sensor cache here to
111 // ensure the objects all exist?
112
113 // Polling loop.
Brad Bishopd499ca62016-12-19 09:24:50 -0500114 while (!_shutdown)
Brad Bishope55ef3d2016-12-19 09:12:40 -0500115 {
116 // Iterate through all the sensors.
Brad Bishop75b4ab82017-01-06 09:33:50 -0500117 for (auto& i : state)
Brad Bishope55ef3d2016-12-19 09:12:40 -0500118 {
Brad Bishop75b4ab82017-01-06 09:33:50 -0500119 auto& attrs = std::get<0>(i.second);
120 if (attrs.find(hwmon::entry::input) != attrs.end())
Brad Bishope55ef3d2016-12-19 09:12:40 -0500121 {
122 // Read value from sensor.
123 int value = 0;
Brad Bishopd499ca62016-12-19 09:24:50 -0500124 read_sysfs(make_sysfs_path(_path,
Brad Bishope55ef3d2016-12-19 09:12:40 -0500125 i.first.first, i.first.second,
126 hwmon::entry::input),
127 value);
128
129 // Update sensor cache.
130 if (sensor_cache->update(i.first, value))
131 {
132 // TODO: Issue#4 - dbus event here.
133 std::cout << i.first.first << i.first.second << " = "
134 << value << std::endl;
135 }
136 }
137 }
138
Brad Bishop9c7b6e02016-12-19 12:43:36 -0500139 // Respond to DBus
140 _bus.process_discard();
141
Brad Bishope55ef3d2016-12-19 09:12:40 -0500142 // Sleep until next interval.
143 // TODO: Issue#5 - Make this configurable.
144 // TODO: Issue#6 - Optionally look at polling interval sysfs entry.
Brad Bishop9c7b6e02016-12-19 12:43:36 -0500145 _bus.wait((1000000us).count());
Brad Bishope55ef3d2016-12-19 09:12:40 -0500146
147 // TODO: Issue#7 - Should probably periodically check the SensorSet
148 // for new entries.
149 }
Brad Bishope55ef3d2016-12-19 09:12:40 -0500150}
151
152// vim: tabstop=8 expandtab shiftwidth=4 softtabstop=4