blob: 0c278ab1115a4ca1b77990334f95185182bd91b6 [file] [log] [blame]
/**
* Copyright © 2016 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 <iostream>
#include <exception>
#include <chrono>
#include <algorithm>
#include <phosphor-logging/log.hpp>
#include "manager.hpp"
#include "errors.hpp"
using namespace std::literals::chrono_literals;
namespace phosphor
{
namespace inventory
{
namespace manager
{
/** @brief Fowrarding signal callback.
*
* Extracts per-signal specific context and forwards the call to the manager
* instance.
*/
auto _signal(sd_bus_message* m, void* data, sd_bus_error* e) noexcept
{
try
{
auto msg = sdbusplus::message::message(m);
auto& args = *static_cast<Manager::SigArg*>(data);
sd_bus_message_ref(m);
auto& mgr = *std::get<0>(args);
mgr.handleEvent(
msg,
static_cast<const DbusSignal&>(
*std::get<1>(args)),
*std::get<2>(args));
}
catch (const std::exception& e)
{
std::cerr << e.what() << std::endl;
}
return 0;
}
Manager::Manager(
sdbusplus::bus::bus&& bus,
const char* busname,
const char* root,
const char* iface) :
ServerObject<ManagerIface>(bus, root),
_shutdown(false),
_root(root),
_bus(std::move(bus)),
_manager(sdbusplus::server::manager::manager(_bus, root))
{
for (auto& group : _events)
{
for (auto pEvent : std::get<std::vector<EventBasePtr>>(
group))
{
if (pEvent->type !=
Event::Type::DBUS_SIGNAL)
{
continue;
}
// Create a callback context for this event group.
auto dbusEvent = static_cast<DbusSignal*>(
pEvent.get());
// Go ahead and store an iterator pointing at
// the event data to avoid lookups later since
// additional signal callbacks aren't added
// after the manager is constructed.
_sigargs.emplace_back(
std::make_unique<SigArg>(
this,
dbusEvent,
&group));
// Register our callback and the context for
// each signal event.
_matches.emplace_back(
_bus,
dbusEvent->signature,
_signal,
_sigargs.back().get());
}
}
_bus.request_name(busname);
}
void Manager::shutdown() noexcept
{
_shutdown = true;
}
void Manager::run() noexcept
{
sdbusplus::message::message unusedMsg{nullptr};
// Run startup events.
for (auto& group : _events)
{
for (auto pEvent : std::get<std::vector<EventBasePtr>>(
group))
{
if (pEvent->type ==
Event::Type::STARTUP)
{
handleEvent(unusedMsg, *pEvent, group);
}
}
}
while (!_shutdown)
{
try
{
_bus.process_discard();
_bus.wait((5000000us).count());
}
catch (const std::exception& e)
{
std::cerr << e.what() << std::endl;
}
}
}
void Manager::updateInterfaces(
const sdbusplus::message::object_path& path,
const Object& interfaces,
ObjectReferences::iterator pos,
bool newObject)
{
auto& refaces = pos->second;
auto ifaceit = interfaces.cbegin();
auto opsit = _makers.cbegin();
auto refaceit = refaces.begin();
std::vector<std::string> signals;
while (ifaceit != interfaces.cend())
{
try
{
// Find the binding ops for this interface.
opsit = std::lower_bound(
opsit,
_makers.cend(),
ifaceit->first,
compareFirst(_makers.key_comp()));
if (opsit == _makers.cend() || opsit->first != ifaceit->first)
{
// This interface is not supported.
throw InterfaceError(
"Encountered unsupported interface.",
ifaceit->first);
}
// Find the binding insertion point or the binding to update.
refaceit = std::lower_bound(
refaceit,
refaces.end(),
ifaceit->first,
compareFirst(refaces.key_comp()));
if (refaceit == refaces.end() || refaceit->first != ifaceit->first)
{
// Add the new interface.
auto& ctor = std::get<MakerType>(opsit->second);
refaceit = refaces.insert(
refaceit,
std::make_pair(
ifaceit->first,
ctor(
_bus,
path.str.c_str(),
ifaceit->second)));
signals.push_back(ifaceit->first);
}
else
{
// Set the new property values.
auto& assign = std::get<AssignerType>(opsit->second);
assign(ifaceit->second, refaceit->second);
}
}
catch (const InterfaceError& e)
{
// Reset the binding ops iterator since we are
// at the end.
opsit = _makers.cbegin();
e.log();
}
++ifaceit;
}
if (newObject)
{
_bus.emit_object_added(path.str.c_str());
}
else if (!signals.empty())
{
// TODO: emit an interfaces added signal
}
}
void Manager::updateObjects(
const std::map<sdbusplus::message::object_path, Object>& objs)
{
auto objit = objs.cbegin();
auto refit = _refs.begin();
std::string absPath;
bool newObj;
while (objit != objs.cend())
{
// Find the insertion point or the object to update.
refit = std::lower_bound(
refit,
_refs.end(),
objit->first,
compareFirst(RelPathCompare(_root)));
absPath.assign(_root);
absPath.append(objit->first);
newObj = false;
if (refit == _refs.end() || refit->first != absPath)
{
refit = _refs.insert(
refit,
std::make_pair(
absPath,
decltype(_refs)::mapped_type()));
newObj = true;
}
updateInterfaces(absPath, objit->second, refit, newObj);
++objit;
}
}
void Manager::notify(std::map<sdbusplus::message::object_path, Object> objs)
{
updateObjects(objs);
}
void Manager::handleEvent(
sdbusplus::message::message& msg,
const Event& event,
const EventInfo& info)
{
auto& actions = std::get<1>(info);
for (auto& f : event)
{
if (!f(_bus, msg, *this))
{
return;
}
}
for (auto& action : actions)
{
action(_bus, *this);
}
}
void Manager::destroyObjects(
const std::vector<const char*>& paths)
{
std::string p;
for (const auto& path : paths)
{
p.assign(_root);
p.append(path);
_bus.emit_object_removed(p.c_str());
_refs.erase(p);
}
}
void Manager::createObjects(
const std::map<sdbusplus::message::object_path, Object>& objs)
{
updateObjects(objs);
}
any_ns::any& Manager::getInterfaceHolder(
const char* path, const char* interface)
{
return const_cast<any_ns::any&>(
const_cast<const Manager*>(
this)->getInterfaceHolder(path, interface));
}
const any_ns::any& Manager::getInterfaceHolder(
const char* path, const char* interface) const
{
std::string p{path};
auto oit = _refs.find(_root + p);
if (oit == _refs.end())
throw std::runtime_error(
_root + p + " was not found");
auto& obj = oit->second;
auto iit = obj.find(interface);
if (iit == obj.end())
throw std::runtime_error(
"interface was not found");
return iit->second;
}
} // namespace manager
} // namespace inventory
} // namespace phosphor
// vim: tabstop=8 expandtab shiftwidth=4 softtabstop=4