-
Notifications
You must be signed in to change notification settings - Fork 74
/
host-interface.cpp
122 lines (95 loc) · 3.84 KB
/
host-interface.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
#include "config.h"
#include "host-interface.hpp"
#include "systemintfcmds.hpp"
#include <ipmid-host/cmd-utils.hpp>
#include <ipmid-host/cmd.hpp>
#include <ipmid/api.hpp>
#include <ipmid/utils.hpp>
#include <phosphor-logging/lg2.hpp>
#include <functional>
#include <memory>
#include <optional>
namespace phosphor
{
namespace host
{
namespace command
{
// When you see Base:: you know we're referencing our base class
namespace Base = sdbusplus::server::xyz::openbmc_project::control;
// IPMI OEM command.
// https://github.com/openbmc/openbmc/issues/2082 for handling
// Non-OEM commands that need to send SMS_ATN
using OEMCmd = uint8_t;
// Map of IPMI OEM command to its equivalent interface command.
// This is needed when invoking the callback handler to indicate
// the status of the executed command.
static const std::map<OEMCmd, Host::Command> intfCommand = {
{CMD_HEARTBEAT, Base::Host::Command::Heartbeat},
{CMD_POWER, Base::Host::Command::SoftOff}};
// Map of Interface command to its corresponding IPMI OEM command.
// This is needed when pushing IPMI commands to command manager's
// queue. The same pair will be returned when IPMI asks us
// why a SMS_ATN was sent
static const std::map<Host::Command, IpmiCmdData> ipmiCommand = {
{Base::Host::Command::Heartbeat, std::make_pair(CMD_HEARTBEAT, 0x00)},
{Base::Host::Command::SoftOff, std::make_pair(CMD_POWER, SOFT_OFF)}};
// Called at user request
void Host::execute(Base::Host::Command command)
{
lg2::debug("Pushing cmd on to queue, control host cmd: {CONTROL_HOST_CMD}",
"CONTROL_HOST_CMD", convertForMessage(command));
auto cmd = std::make_tuple(
ipmiCommand.at(command),
std::bind(&Host::commandStatusHandler, this, std::placeholders::_1,
std::placeholders::_2));
ipmid_send_cmd_to_host(std::move(cmd));
}
// Called into by Command Manager
void Host::commandStatusHandler(IpmiCmdData cmd, bool status)
{
// Need to convert <cmd> to the equivalent one mentioned in spec
auto value = status ? Result::Success : Result::Failure;
// Fire a signal
this->commandComplete(intfCommand.at(std::get<0>(cmd)), value);
}
Host::FirmwareCondition Host::currentFirmwareCondition() const
{
// shared object used to wait for host response
auto hostCondition =
std::make_shared<std::optional<Host::FirmwareCondition>>();
// callback for command to host
auto hostAckCallback = [hostCondition](IpmiCmdData, bool status) {
auto value = status ? Host::FirmwareCondition::Running
: Host::FirmwareCondition::Off;
lg2::debug("currentFirmwareCondition:hostAckCallback fired, "
"control host cmd: {CONTROL_HOST_CMD}",
"CONTROL_HOST_CMD", value);
*(hostCondition.get()) = value;
return;
};
auto cmd = phosphor::host::command::CommandHandler(
ipmiCommand.at(Base::Host::Command::Heartbeat),
std::move(hostAckCallback));
ipmid_send_cmd_to_host(std::move(cmd));
// Timer to ensure this function returns something within a reasonable time
sdbusplus::Timer hostAckTimer([hostCondition]() {
lg2::debug("currentFirmwareCondition: timer expired!");
*(hostCondition.get()) = Host::FirmwareCondition::Off;
});
// Wait 1 second past the ATN_ACK timeout to ensure we wait for as
// long as the timeout
hostAckTimer.start(std::chrono::seconds(IPMI_SMS_ATN_ACK_TIMEOUT_SECS + 1));
auto io = getIoContext();
while (!hostCondition.get()->has_value())
{
lg2::debug("currentFirmwareCondition: waiting for host response");
io->run_for(std::chrono::milliseconds(100));
}
hostAckTimer.stop();
lg2::debug("currentFirmwareCondition: hostCondition is ready!");
return hostCondition.get()->value();
}
} // namespace command
} // namespace host
} // namespace phosphor