blob: 74ce8e96f51571388434e15a3d9f126b10d38609 [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"
4#include "session.hpp"
5#include "sol_context.hpp"
6
Vernon Mauery7e4a6512018-11-09 08:43:36 -08007#include <boost/asio/io_context.hpp>
8#include <boost/asio/local/stream_protocol.hpp>
Andrew Geissler9d9b7632020-05-17 09:18:05 -05009#include <cstddef>
Vernon Mauery9e801a22018-10-12 13:20:49 -070010#include <map>
11#include <memory>
Andrew Geissler7408e762020-05-17 08:56:05 -050012#include <string>
Vernon Mauery9e801a22018-10-12 13:20:49 -070013
Tom Joseph5a454a22017-02-15 14:51:42 +053014namespace sol
15{
16
Tom Joseph5a454a22017-02-15 14:51:42 +053017constexpr size_t MAX_PAYLOAD_SIZE = 255;
18constexpr uint8_t MAJOR_VERSION = 0x01;
19constexpr uint8_t MINOR_VERSION = 0x00;
20
21constexpr char CONSOLE_SOCKET_PATH[] = "\0obmc-console";
22constexpr size_t CONSOLE_SOCKET_PATH_LEN = sizeof(CONSOLE_SOCKET_PATH) - 1;
23
Tom Joseph1967cf52017-04-28 01:24:22 +053024constexpr uint8_t accIntervalFactor = 5;
25constexpr uint8_t retryIntervalFactor = 10;
26
Tom Joseph5a454a22017-02-15 14:51:42 +053027using Instance = uint8_t;
28
Tom Joseph704f3422017-04-26 10:13:03 +053029using namespace std::chrono_literals;
Tom Joseph5a454a22017-02-15 14:51:42 +053030
31/** @class Manager
32 *
33 * Manager class acts a manager for the SOL payload instances and provides
34 * interfaces to start a payload instance, stop a payload instance and get
35 * reference to the context object.
36 */
37class Manager
38{
Vernon Mauery9e801a22018-10-12 13:20:49 -070039 public:
40 /** @brief SOL Payload Instance is the key for the map, the value is the
41 * SOL context.
42 */
Vernon Mauerya6ad5e12020-02-21 14:54:24 -080043 using SOLPayloadMap = std::map<Instance, std::shared_ptr<Context>>;
Tom Joseph5a454a22017-02-15 14:51:42 +053044
Vernon Mauery7e4a6512018-11-09 08:43:36 -080045 Manager() = delete;
Vernon Mauery9e801a22018-10-12 13:20:49 -070046 ~Manager() = default;
47 Manager(const Manager&) = delete;
48 Manager& operator=(const Manager&) = delete;
49 Manager(Manager&&) = default;
50 Manager& operator=(Manager&&) = default;
Tom Joseph5a454a22017-02-15 14:51:42 +053051
Vernon Mauery7e4a6512018-11-09 08:43:36 -080052 Manager(std::shared_ptr<boost::asio::io_context> io) : io(io)
53 {
54 }
55
56 /** @brief io context to add events to */
57 std::shared_ptr<boost::asio::io_context> io;
58
Vernon Mauery9e801a22018-10-12 13:20:49 -070059 /** @brief Host Console Buffer. */
60 ConsoleData dataBuffer;
Tom Joseph5a454a22017-02-15 14:51:42 +053061
Vernon Mauery9e801a22018-10-12 13:20:49 -070062 /** @brief Set in Progress.
63 *
64 * This parameter is used to indicate when any of the SOL parameters
65 * are being updated, and when the changes are completed. The bit is
66 * primarily provided to alert software than some other software or
67 * utility is in the process of making changes to the data. This field
68 * is initialized to set complete.
69 */
70 uint8_t progress = 0;
Tom Joseph5a454a22017-02-15 14:51:42 +053071
Vernon Mauery9e801a22018-10-12 13:20:49 -070072 /** @brief SOL enable
73 *
74 * This controls whether the SOL payload can be activated. By default
75 * the SOL is enabled.
76 */
77 bool enable = true;
Tom Joseph1967cf52017-04-28 01:24:22 +053078
Vernon Mauery9e801a22018-10-12 13:20:49 -070079 /** @brief SOL payload encryption.
80 *
81 * Force encryption: if the cipher suite for the session supports
82 * encryption, then this setting will force the use of encryption for
83 * all SOL payload data. Encryption controlled by remote console:
84 * Whether SOL packets are encrypted or not is selectable by the remote
85 * console at the time the payload is activated. The default is force
86 * encryption.
87 */
88 bool forceEncrypt = true;
Tom Joseph1967cf52017-04-28 01:24:22 +053089
Vernon Mauery9e801a22018-10-12 13:20:49 -070090 /** @brief SOL payload authentication.
91 *
92 * Force authentication: if the cipher suite for the session supports
93 * authentication, then this setting will force the use of for
94 * authentication for all SOL payload data. Authentication controlled
95 * by remote console: Note that for the standard Cipher Suites,
96 * if encryption is used authentication must also be used. Therefore,
97 * while encryption is being used software will not be able to select
98 * using unauthenticated payloads.
99 */
100 bool forceAuth = true;
Tom Joseph1967cf52017-04-28 01:24:22 +0530101
Vernon Mauery9e801a22018-10-12 13:20:49 -0700102 /** @brief SOL privilege level.
103 *
104 * Sets the minimum operating privilege level that is required to be
105 * able to activate SOL using the Activate Payload command.
106 */
107 session::Privilege solMinPrivilege = session::Privilege::USER;
Tom Joseph1967cf52017-04-28 01:24:22 +0530108
Vernon Mauery9e801a22018-10-12 13:20:49 -0700109 /** @brief Character Accumulate Interval
110 *
111 * This sets the typical amount of time that the BMC will wait before
112 * transmitting a partial SOL character data packet. (Where a partial
113 * packet is defined as a packet that has fewer characters to transmit
114 * than the number of characters specified by the character send
115 * threshold. This parameter can be modified by the set SOL
116 * configuration parameters command. The SOL configuration parameter,
117 * Character Accumulate Interval is 5 ms increments, 1-based value. The
118 * parameter value is accumulateInterval/5. The accumulateInterval
119 * needs to be a multiple of 5.
120 */
121 std::chrono::milliseconds accumulateInterval = 100ms;
Tom Joseph1967cf52017-04-28 01:24:22 +0530122
Vernon Mauery9e801a22018-10-12 13:20:49 -0700123 /** @brief Character Send Threshold
124 *
125 * The BMC will automatically send an SOL character data packet
126 * containing this number of characters as soon as this number of
127 * characters (or greater) has been accepted from the baseboard serial
128 * controller into the BMC. This provides a mechanism to tune the
129 * buffer to reduce latency to when the first characters are received
130 * after an idle interval. In the degenerate case, setting this value
131 * to a ‘1’ would cause the BMC to send a packet as soon as the first
132 * character was received. This parameter can be modified by the set
133 * SOL configuration parameters command.
134 */
135 uint8_t sendThreshold = 1;
Tom Joseph5a454a22017-02-15 14:51:42 +0530136
Vernon Mauery9e801a22018-10-12 13:20:49 -0700137 /** @brief Retry Count
138 *
139 * 1-based. 0 = no retries after packet is transmitted. Packet will be
140 * dropped if no ACK/NACK received by time retries expire. The maximum
141 * value for retry count is 7. This parameter can be modified by the
142 * set SOL configuration parameters command.
143 */
144 uint8_t retryCount = 7;
Tom Joseph5a454a22017-02-15 14:51:42 +0530145
Vernon Mauery9e801a22018-10-12 13:20:49 -0700146 /** @brief Retry Interval
147 *
148 * Sets the time that the BMC will wait before the first retry and the
149 * time between retries when sending SOL packets to the remote console.
150 * This parameter can be modified by the set SOL configuration
151 * parameters command. The SOL configuration parameter Retry Interval
152 * is 10 ms increments, 1-based value. The parameter value is
153 * retryInterval/10. The retryInterval needs to be a multiple of 10.
154 */
155 std::chrono::milliseconds retryInterval = 100ms;
Tom Joseph5a454a22017-02-15 14:51:42 +0530156
Vernon Mauery9e801a22018-10-12 13:20:49 -0700157 /** @brief Channel Number
158 *
159 * This parameter indicates which IPMI channel is being used for the
160 * communication parameters (e.g. IP address, MAC address) for the SOL
161 * Payload. Typically, these parameters will come from the same channel
162 * that the Activate Payload command for SOL was accepted over. The
163 * network channel number is defaulted to 1.
164 */
165 uint8_t channel = 1;
Tom Joseph5a454a22017-02-15 14:51:42 +0530166
Vernon Mauery7e4a6512018-11-09 08:43:36 -0800167 /** @brief Add host console I/O event source to the event loop. */
168 void startHostConsole();
169
170 /** @brief Remove host console I/O event source. */
171 void stopHostConsole();
172
Vernon Mauery9e801a22018-10-12 13:20:49 -0700173 /** @brief Start a SOL payload instance.
174 *
175 * Starting a payload instance involves creating the context object,
176 * add the accumulate interval timer and retry interval timer to the
177 * event loop.
178 *
179 * @param[in] payloadInstance - SOL payload instance.
180 * @param[in] sessionID - BMC session ID.
181 */
182 void startPayloadInstance(uint8_t payloadInstance,
183 session::SessionID sessionID);
Tom Joseph6f83cbc2018-03-27 03:01:58 +0530184
Vernon Mauery9e801a22018-10-12 13:20:49 -0700185 /** @brief Stop SOL payload instance.
186 *
187 * Stopping a payload instance involves stopping and removing the
188 * accumulate interval timer and retry interval timer from the event
189 * loop, delete the context object.
190 *
191 * @param[in] payloadInstance - SOL payload instance
192 */
193 void stopPayloadInstance(uint8_t payloadInstance);
Tom Joseph5a454a22017-02-15 14:51:42 +0530194
Vernon Mauery9e801a22018-10-12 13:20:49 -0700195 /** @brief Get SOL Context by Payload Instance.
196 *
197 * @param[in] payloadInstance - SOL payload instance.
198 *
199 * @return reference to the SOL payload context.
200 */
201 Context& getContext(uint8_t payloadInstance)
202 {
203 auto iter = payloadMap.find(payloadInstance);
Tom Joseph5a454a22017-02-15 14:51:42 +0530204
Vernon Mauery9e801a22018-10-12 13:20:49 -0700205 if (iter != payloadMap.end())
Tom Joseph5a454a22017-02-15 14:51:42 +0530206 {
Vernon Mauery9e801a22018-10-12 13:20:49 -0700207 return *(iter->second);
Tom Joseph5a454a22017-02-15 14:51:42 +0530208 }
209
Vernon Mauery9e801a22018-10-12 13:20:49 -0700210 std::string msg = "Invalid SOL payload instance " + payloadInstance;
211 throw std::runtime_error(msg.c_str());
212 }
213
214 /** @brief Get SOL Context by Session ID.
215 *
216 * @param[in] sessionID - IPMI Session ID.
217 *
218 * @return reference to the SOL payload context.
219 */
220 Context& getContext(session::SessionID sessionID)
221 {
222 for (const auto& kv : payloadMap)
Tom Joseph5a454a22017-02-15 14:51:42 +0530223 {
Vernon Mauery9e801a22018-10-12 13:20:49 -0700224 if (kv.second->sessionID == sessionID)
225 {
226 return *kv.second;
227 }
Tom Joseph5a454a22017-02-15 14:51:42 +0530228 }
229
Vernon Mauery9e801a22018-10-12 13:20:49 -0700230 std::string msg = "Invalid SOL SessionID " + sessionID;
231 throw std::runtime_error(msg.c_str());
232 }
Tom Joseph5a454a22017-02-15 14:51:42 +0530233
Vernon Mauery9e801a22018-10-12 13:20:49 -0700234 /** @brief Check if SOL payload is active.
235 *
236 * @param[in] payloadInstance - SOL payload instance.
237 *
238 * @return true if the instance is active and false it is not active.
239 */
240 auto isPayloadActive(uint8_t payloadInstance) const
241 {
242 return (0 != payloadMap.count(payloadInstance));
243 }
Tom Joseph5a454a22017-02-15 14:51:42 +0530244
Vernon Mauery9e801a22018-10-12 13:20:49 -0700245 /** @brief Write data to the host console unix socket.
246 *
247 * @param[in] input - Data from the remote console.
248 *
249 * @return 0 on success and errno on failure.
250 */
251 int writeConsoleSocket(const std::vector<uint8_t>& input) const;
Tom Joseph5a454a22017-02-15 14:51:42 +0530252
Vernon Mauery9e801a22018-10-12 13:20:49 -0700253 private:
254 SOLPayloadMap payloadMap;
255
Vernon Mauery7e4a6512018-11-09 08:43:36 -0800256 /** @brief Local stream socket for the host console. */
257 std::unique_ptr<boost::asio::local::stream_protocol::socket> consoleSocket =
258 nullptr;
Vernon Mauery9e801a22018-10-12 13:20:49 -0700259
260 /** @brief Initialize the host console file descriptor. */
Vernon Mauery7e4a6512018-11-09 08:43:36 -0800261 void initConsoleSocket();
262
263 /** @brief Handle incoming console data on the console socket */
264 void consoleInputHandler();
Tom Joseph5a454a22017-02-15 14:51:42 +0530265};
266
Vernon Mauery9e801a22018-10-12 13:20:49 -0700267} // namespace sol