blob: 72c13b42e7b63ccb1029bfe7915c49470f2f12bd [file] [log] [blame]
Tom Joseph5a454a22017-02-15 14:51:42 +05301#pragma once
2
Tom Joseph5a454a22017-02-15 14:51:42 +05303#include "console_buffer.hpp"
Vernon Mauery2085ae02021-06-10 11:51:00 -07004#include "main.hpp"
Tom Joseph5a454a22017-02-15 14:51:42 +05305#include "session.hpp"
6#include "sol_context.hpp"
7
Vernon Mauery7e4a6512018-11-09 08:43:36 -08008#include <boost/asio/io_context.hpp>
9#include <boost/asio/local/stream_protocol.hpp>
George Liubc8958f2022-07-04 09:29:49 +080010
Andrew Geissler9d9b7632020-05-17 09:18:05 -050011#include <cstddef>
Vernon Mauery9e801a22018-10-12 13:20:49 -070012#include <map>
13#include <memory>
Andrew Geissler7408e762020-05-17 08:56:05 -050014#include <string>
Vernon Mauery9e801a22018-10-12 13:20:49 -070015
Tom Joseph5a454a22017-02-15 14:51:42 +053016namespace sol
17{
18
Tom Joseph5a454a22017-02-15 14:51:42 +053019constexpr size_t MAX_PAYLOAD_SIZE = 255;
20constexpr uint8_t MAJOR_VERSION = 0x01;
21constexpr uint8_t MINOR_VERSION = 0x00;
22
23constexpr char CONSOLE_SOCKET_PATH[] = "\0obmc-console";
24constexpr size_t CONSOLE_SOCKET_PATH_LEN = sizeof(CONSOLE_SOCKET_PATH) - 1;
25
Tom Joseph1967cf52017-04-28 01:24:22 +053026constexpr uint8_t accIntervalFactor = 5;
27constexpr uint8_t retryIntervalFactor = 10;
28
Tom Joseph5a454a22017-02-15 14:51:42 +053029using Instance = uint8_t;
30
Tom Joseph704f3422017-04-26 10:13:03 +053031using namespace std::chrono_literals;
Tom Joseph5a454a22017-02-15 14:51:42 +053032
33/** @class Manager
34 *
35 * Manager class acts a manager for the SOL payload instances and provides
36 * interfaces to start a payload instance, stop a payload instance and get
37 * reference to the context object.
38 */
39class Manager
40{
Vernon Mauery2085ae02021-06-10 11:51:00 -070041 private:
42 struct Private
George Liubc8958f2022-07-04 09:29:49 +080043 {};
Vernon Mauery2085ae02021-06-10 11:51:00 -070044
Vernon Mauery9e801a22018-10-12 13:20:49 -070045 public:
46 /** @brief SOL Payload Instance is the key for the map, the value is the
47 * SOL context.
48 */
Vernon Mauerya6ad5e12020-02-21 14:54:24 -080049 using SOLPayloadMap = std::map<Instance, std::shared_ptr<Context>>;
Tom Joseph5a454a22017-02-15 14:51:42 +053050
Vernon Mauery7e4a6512018-11-09 08:43:36 -080051 Manager() = delete;
Vernon Mauery9e801a22018-10-12 13:20:49 -070052 ~Manager() = default;
53 Manager(const Manager&) = delete;
54 Manager& operator=(const Manager&) = delete;
55 Manager(Manager&&) = default;
56 Manager& operator=(Manager&&) = default;
Tom Joseph5a454a22017-02-15 14:51:42 +053057
Vernon Mauery2085ae02021-06-10 11:51:00 -070058 Manager(std::shared_ptr<boost::asio::io_context>& io, const Private&) :
59 io(io)
George Liubc8958f2022-07-04 09:29:49 +080060 {}
Vernon Mauery7e4a6512018-11-09 08:43:36 -080061
Vernon Mauery2085ae02021-06-10 11:51:00 -070062 /**
63 * @brief Get a reference to the singleton Manager
64 *
65 * @return Manager reference
66 */
67 static Manager& get()
68 {
69 static std::shared_ptr<Manager> ptr = nullptr;
70 if (!ptr)
71 {
72 std::shared_ptr<boost::asio::io_context> io = getIo();
73 ptr = std::make_shared<Manager>(io, Private());
74 }
75 return *ptr;
76 }
77
Vernon Mauery7e4a6512018-11-09 08:43:36 -080078 /** @brief io context to add events to */
79 std::shared_ptr<boost::asio::io_context> io;
80
Vernon Mauery9e801a22018-10-12 13:20:49 -070081 /** @brief Host Console Buffer. */
82 ConsoleData dataBuffer;
Tom Joseph5a454a22017-02-15 14:51:42 +053083
Vernon Mauery9e801a22018-10-12 13:20:49 -070084 /** @brief Set in Progress.
85 *
86 * This parameter is used to indicate when any of the SOL parameters
87 * are being updated, and when the changes are completed. The bit is
88 * primarily provided to alert software than some other software or
89 * utility is in the process of making changes to the data. This field
90 * is initialized to set complete.
91 */
92 uint8_t progress = 0;
Tom Joseph5a454a22017-02-15 14:51:42 +053093
Vernon Mauery9e801a22018-10-12 13:20:49 -070094 /** @brief SOL enable
95 *
96 * This controls whether the SOL payload can be activated. By default
97 * the SOL is enabled.
98 */
99 bool enable = true;
Tom Joseph1967cf52017-04-28 01:24:22 +0530100
Vernon Mauery9e801a22018-10-12 13:20:49 -0700101 /** @brief SOL payload encryption.
102 *
103 * Force encryption: if the cipher suite for the session supports
104 * encryption, then this setting will force the use of encryption for
105 * all SOL payload data. Encryption controlled by remote console:
106 * Whether SOL packets are encrypted or not is selectable by the remote
107 * console at the time the payload is activated. The default is force
108 * encryption.
109 */
110 bool forceEncrypt = true;
Tom Joseph1967cf52017-04-28 01:24:22 +0530111
Vernon Mauery9e801a22018-10-12 13:20:49 -0700112 /** @brief SOL payload authentication.
113 *
114 * Force authentication: if the cipher suite for the session supports
115 * authentication, then this setting will force the use of for
116 * authentication for all SOL payload data. Authentication controlled
117 * by remote console: Note that for the standard Cipher Suites,
118 * if encryption is used authentication must also be used. Therefore,
119 * while encryption is being used software will not be able to select
120 * using unauthenticated payloads.
121 */
122 bool forceAuth = true;
Tom Joseph1967cf52017-04-28 01:24:22 +0530123
Vernon Mauery9e801a22018-10-12 13:20:49 -0700124 /** @brief SOL privilege level.
125 *
126 * Sets the minimum operating privilege level that is required to be
127 * able to activate SOL using the Activate Payload command.
128 */
129 session::Privilege solMinPrivilege = session::Privilege::USER;
Tom Joseph1967cf52017-04-28 01:24:22 +0530130
Vernon Mauery9e801a22018-10-12 13:20:49 -0700131 /** @brief Character Accumulate Interval
132 *
133 * This sets the typical amount of time that the BMC will wait before
134 * transmitting a partial SOL character data packet. (Where a partial
135 * packet is defined as a packet that has fewer characters to transmit
136 * than the number of characters specified by the character send
137 * threshold. This parameter can be modified by the set SOL
138 * configuration parameters command. The SOL configuration parameter,
139 * Character Accumulate Interval is 5 ms increments, 1-based value. The
140 * parameter value is accumulateInterval/5. The accumulateInterval
141 * needs to be a multiple of 5.
142 */
143 std::chrono::milliseconds accumulateInterval = 100ms;
Tom Joseph1967cf52017-04-28 01:24:22 +0530144
Vernon Mauery9e801a22018-10-12 13:20:49 -0700145 /** @brief Character Send Threshold
146 *
147 * The BMC will automatically send an SOL character data packet
148 * containing this number of characters as soon as this number of
149 * characters (or greater) has been accepted from the baseboard serial
150 * controller into the BMC. This provides a mechanism to tune the
151 * buffer to reduce latency to when the first characters are received
152 * after an idle interval. In the degenerate case, setting this value
153 * to a ‘1’ would cause the BMC to send a packet as soon as the first
154 * character was received. This parameter can be modified by the set
155 * SOL configuration parameters command.
156 */
157 uint8_t sendThreshold = 1;
Tom Joseph5a454a22017-02-15 14:51:42 +0530158
Vernon Mauery9e801a22018-10-12 13:20:49 -0700159 /** @brief Retry Count
160 *
161 * 1-based. 0 = no retries after packet is transmitted. Packet will be
162 * dropped if no ACK/NACK received by time retries expire. The maximum
163 * value for retry count is 7. This parameter can be modified by the
164 * set SOL configuration parameters command.
165 */
166 uint8_t retryCount = 7;
Tom Joseph5a454a22017-02-15 14:51:42 +0530167
Vernon Mauery9e801a22018-10-12 13:20:49 -0700168 /** @brief Retry Interval
169 *
170 * Sets the time that the BMC will wait before the first retry and the
171 * time between retries when sending SOL packets to the remote console.
172 * This parameter can be modified by the set SOL configuration
173 * parameters command. The SOL configuration parameter Retry Interval
174 * is 10 ms increments, 1-based value. The parameter value is
175 * retryInterval/10. The retryInterval needs to be a multiple of 10.
176 */
177 std::chrono::milliseconds retryInterval = 100ms;
Tom Joseph5a454a22017-02-15 14:51:42 +0530178
Vernon Mauery9e801a22018-10-12 13:20:49 -0700179 /** @brief Channel Number
180 *
181 * This parameter indicates which IPMI channel is being used for the
182 * communication parameters (e.g. IP address, MAC address) for the SOL
183 * Payload. Typically, these parameters will come from the same channel
184 * that the Activate Payload command for SOL was accepted over. The
185 * network channel number is defaulted to 1.
186 */
187 uint8_t channel = 1;
Tom Joseph5a454a22017-02-15 14:51:42 +0530188
Vernon Mauery7e4a6512018-11-09 08:43:36 -0800189 /** @brief Add host console I/O event source to the event loop. */
190 void startHostConsole();
191
192 /** @brief Remove host console I/O event source. */
193 void stopHostConsole();
194
Vernon Mauery9e801a22018-10-12 13:20:49 -0700195 /** @brief Start a SOL payload instance.
196 *
197 * Starting a payload instance involves creating the context object,
198 * add the accumulate interval timer and retry interval timer to the
199 * event loop.
200 *
201 * @param[in] payloadInstance - SOL payload instance.
202 * @param[in] sessionID - BMC session ID.
203 */
204 void startPayloadInstance(uint8_t payloadInstance,
205 session::SessionID sessionID);
Tom Joseph6f83cbc2018-03-27 03:01:58 +0530206
Vernon Mauery9e801a22018-10-12 13:20:49 -0700207 /** @brief Stop SOL payload instance.
208 *
209 * Stopping a payload instance involves stopping and removing the
210 * accumulate interval timer and retry interval timer from the event
211 * loop, delete the context object.
212 *
213 * @param[in] payloadInstance - SOL payload instance
214 */
215 void stopPayloadInstance(uint8_t payloadInstance);
Tom Joseph5a454a22017-02-15 14:51:42 +0530216
srikanta mondalf6e72302020-06-08 12:46:28 +0000217 /* @brief Stop all the active SOL payload instances */
218 void stopAllPayloadInstance();
219
Vernon Mauery9e801a22018-10-12 13:20:49 -0700220 /** @brief Get SOL Context by Payload Instance.
221 *
222 * @param[in] payloadInstance - SOL payload instance.
223 *
224 * @return reference to the SOL payload context.
225 */
226 Context& getContext(uint8_t payloadInstance)
227 {
228 auto iter = payloadMap.find(payloadInstance);
Tom Joseph5a454a22017-02-15 14:51:42 +0530229
Vernon Mauery9e801a22018-10-12 13:20:49 -0700230 if (iter != payloadMap.end())
Tom Joseph5a454a22017-02-15 14:51:42 +0530231 {
Vernon Mauery9e801a22018-10-12 13:20:49 -0700232 return *(iter->second);
Tom Joseph5a454a22017-02-15 14:51:42 +0530233 }
234
Vernon Mauery9e801a22018-10-12 13:20:49 -0700235 std::string msg = "Invalid SOL payload instance " + payloadInstance;
236 throw std::runtime_error(msg.c_str());
237 }
238
239 /** @brief Get SOL Context by Session ID.
240 *
241 * @param[in] sessionID - IPMI Session ID.
242 *
243 * @return reference to the SOL payload context.
244 */
245 Context& getContext(session::SessionID sessionID)
246 {
247 for (const auto& kv : payloadMap)
Tom Joseph5a454a22017-02-15 14:51:42 +0530248 {
Vernon Mauery9e801a22018-10-12 13:20:49 -0700249 if (kv.second->sessionID == sessionID)
250 {
251 return *kv.second;
252 }
Tom Joseph5a454a22017-02-15 14:51:42 +0530253 }
254
Vernon Mauery9e801a22018-10-12 13:20:49 -0700255 std::string msg = "Invalid SOL SessionID " + sessionID;
256 throw std::runtime_error(msg.c_str());
257 }
Tom Joseph5a454a22017-02-15 14:51:42 +0530258
Vernon Mauery9e801a22018-10-12 13:20:49 -0700259 /** @brief Check if SOL payload is active.
260 *
261 * @param[in] payloadInstance - SOL payload instance.
262 *
263 * @return true if the instance is active and false it is not active.
264 */
265 auto isPayloadActive(uint8_t payloadInstance) const
266 {
267 return (0 != payloadMap.count(payloadInstance));
268 }
Tom Joseph5a454a22017-02-15 14:51:42 +0530269
Vernon Mauery9e801a22018-10-12 13:20:49 -0700270 /** @brief Write data to the host console unix socket.
271 *
272 * @param[in] input - Data from the remote console.
273 *
274 * @return 0 on success and errno on failure.
275 */
276 int writeConsoleSocket(const std::vector<uint8_t>& input) const;
Cheng C Yang29086952020-03-09 15:29:18 +0800277 void updateSOLParameter(uint8_t channelNum);
Tom Joseph5a454a22017-02-15 14:51:42 +0530278
Vernon Mauery9e801a22018-10-12 13:20:49 -0700279 private:
280 SOLPayloadMap payloadMap;
281
Vernon Mauery7e4a6512018-11-09 08:43:36 -0800282 /** @brief Local stream socket for the host console. */
283 std::unique_ptr<boost::asio::local::stream_protocol::socket> consoleSocket =
284 nullptr;
Vernon Mauery9e801a22018-10-12 13:20:49 -0700285
286 /** @brief Initialize the host console file descriptor. */
Vernon Mauery7e4a6512018-11-09 08:43:36 -0800287 void initConsoleSocket();
288
289 /** @brief Handle incoming console data on the console socket */
290 void consoleInputHandler();
Tom Joseph5a454a22017-02-15 14:51:42 +0530291};
292
srikanta mondalf6e72302020-06-08 12:46:28 +0000293/** @brief Callback method to close SOL sessions for SOL service change */
294void registerSOLServiceChangeCallback();
295
Jian Zhangc936eca2022-03-22 11:25:29 +0800296/** @brief Callback register method to SOL conf parameters change */
297void registerSolConfChangeCallbackHandler(std::string channel);
298
Vernon Mauery9e801a22018-10-12 13:20:49 -0700299} // namespace sol