blob: 87f9ed78bbed67c33f48d8866790c32018bc7e61 [file] [log] [blame]
Vernon Mauery9e801a22018-10-12 13:20:49 -07001#include "sol_manager.hpp"
2
3#include "main.hpp"
4#include "sol_context.hpp"
5
Tom Joseph22c5ad32017-03-14 18:04:22 +05306#include <sys/socket.h>
7#include <sys/un.h>
Vernon Mauery9e801a22018-10-12 13:20:49 -07008
Vernon Mauery7e4a6512018-11-09 08:43:36 -08009#include <boost/asio/basic_stream_socket.hpp>
10#include <boost/asio/io_context.hpp>
11#include <boost/asio/local/stream_protocol.hpp>
12#include <boost/asio/write.hpp>
Tom Joseph2e44e0e2017-03-14 18:09:11 +053013#include <chrono>
Tom Joseph22c5ad32017-03-14 18:04:22 +053014#include <cmath>
15#include <phosphor-logging/log.hpp>
Tom Joseph22c5ad32017-03-14 18:04:22 +053016
17namespace sol
18{
19
20using namespace phosphor::logging;
21
Vernon Mauery7e4a6512018-11-09 08:43:36 -080022void Manager::initConsoleSocket()
Tom Josephb81f7612017-04-25 17:59:02 +053023{
Vernon Mauery7e4a6512018-11-09 08:43:36 -080024 // explicit length constructor for NUL-prefixed abstract path
25 std::string path(CONSOLE_SOCKET_PATH, CONSOLE_SOCKET_PATH_LEN);
26 boost::asio::local::stream_protocol::endpoint ep(path);
27 consoleSocket =
28 std::make_unique<boost::asio::local::stream_protocol::socket>(*io);
29 consoleSocket->connect(ep);
Tom Josephb81f7612017-04-25 17:59:02 +053030}
31
Vernon Mauery7e4a6512018-11-09 08:43:36 -080032void Manager::consoleInputHandler()
Tom Joseph22c5ad32017-03-14 18:04:22 +053033{
Vernon Mauery7e4a6512018-11-09 08:43:36 -080034 boost::system::error_code ec;
35 boost::asio::socket_base::bytes_readable cmd(true);
36 consoleSocket->io_control(cmd, ec);
37 size_t readSize;
38 if (!ec)
Tom Joseph22c5ad32017-03-14 18:04:22 +053039 {
Vernon Mauery7e4a6512018-11-09 08:43:36 -080040 readSize = cmd.get();
41 }
42 else
43 {
44 log<level::ERR>("Reading ready count from host console socket failed:",
45 entry("EXCEPTION=%s", ec.message().c_str()));
46 return;
47 }
48 std::vector<uint8_t> buffer(readSize);
49 ec.clear();
50 size_t readDataLen =
51 consoleSocket->read_some(boost::asio::buffer(buffer), ec);
52 if (ec)
53 {
54 log<level::ERR>("Reading from host console socket failed:",
55 entry("EXCEPTION=%s", ec.message().c_str()));
56 return;
Tom Joseph22c5ad32017-03-14 18:04:22 +053057 }
58
Vernon Mauery7e4a6512018-11-09 08:43:36 -080059 // Update the Console buffer with data read from the socket
60 buffer.resize(readDataLen);
61 dataBuffer.write(buffer);
Tom Joseph22c5ad32017-03-14 18:04:22 +053062}
63
Vernon Mauery70fd29c2017-11-30 13:11:43 -080064int Manager::writeConsoleSocket(const std::vector<uint8_t>& input) const
Tom Josephb51f6412017-03-14 18:08:14 +053065{
Vernon Mauery7e4a6512018-11-09 08:43:36 -080066 boost::system::error_code ec;
67 boost::asio::write(*consoleSocket, boost::asio::buffer(input), ec);
68 return ec.value();
69}
Tom Josephb51f6412017-03-14 18:08:14 +053070
Vernon Mauery7e4a6512018-11-09 08:43:36 -080071void Manager::startHostConsole()
72{
73 if (!consoleSocket)
Tom Josephb51f6412017-03-14 18:08:14 +053074 {
Vernon Mauery7e4a6512018-11-09 08:43:36 -080075 initConsoleSocket();
Tom Josephb51f6412017-03-14 18:08:14 +053076 }
Vernon Mauery7e4a6512018-11-09 08:43:36 -080077 consoleSocket->async_wait(boost::asio::socket_base::wait_read,
78 [this](const boost::system::error_code& ec) {
79 if (!ec)
80 {
81 consoleInputHandler();
82 startHostConsole();
83 }
84 });
Rashmi RV0f63e012019-11-21 15:38:50 +053085} // namespace sol
Tom Josephb51f6412017-03-14 18:08:14 +053086
Vernon Mauery7e4a6512018-11-09 08:43:36 -080087void Manager::stopHostConsole()
88{
89 if (consoleSocket)
90 {
91 consoleSocket->cancel();
92 consoleSocket.reset();
93 }
Tom Josephb51f6412017-03-14 18:08:14 +053094}
95
Tom Joseph2e44e0e2017-03-14 18:09:11 +053096void Manager::startPayloadInstance(uint8_t payloadInstance,
97 session::SessionID sessionID)
98{
99 if (payloadMap.empty())
100 {
Rashmi RV0f63e012019-11-21 15:38:50 +0530101 try
102 {
103 startHostConsole();
104 }
105 catch (const std::exception& e)
106 {
107 log<level::ERR>("Encountered exception when starting host console. "
108 "Hence stopping host console.",
109 entry("EXCEPTION=%s", e.what()));
110 stopHostConsole();
111 throw;
112 }
Tom Joseph2e44e0e2017-03-14 18:09:11 +0530113 }
114
115 // Create the SOL Context data for payload instance
Vernon Mauerya6ad5e12020-02-21 14:54:24 -0800116 std::shared_ptr<Context> context = Context::makeContext(
117 io, retryCount, sendThreshold, payloadInstance, sessionID);
Tom Joseph2e44e0e2017-03-14 18:09:11 +0530118
Tom Joseph2e44e0e2017-03-14 18:09:11 +0530119 payloadMap.emplace(payloadInstance, std::move(context));
120}
121
Tom Joseph4ff14b52017-03-14 18:10:52 +0530122void Manager::stopPayloadInstance(uint8_t payloadInstance)
123{
124 auto iter = payloadMap.find(payloadInstance);
125 if (iter == payloadMap.end())
126 {
127 throw std::runtime_error("SOL Payload instance not found ");
128 }
129
130 payloadMap.erase(iter);
131
Tom Joseph4ff14b52017-03-14 18:10:52 +0530132 if (payloadMap.empty())
133 {
Vernon Mauery7e4a6512018-11-09 08:43:36 -0800134 stopHostConsole();
Tom Josephde203ce2017-08-01 16:48:06 +0530135
136 dataBuffer.erase(dataBuffer.size());
Tom Joseph4ff14b52017-03-14 18:10:52 +0530137 }
138}
139
Tom Joseph22c5ad32017-03-14 18:04:22 +0530140} // namespace sol