xref: /openbmc/phosphor-host-ipmid/include/ipmid/message.hpp (revision b4b409180c2773e1cede1d2e2936881d09639f5a)
1e7329c71SVernon Mauery /**
2e7329c71SVernon Mauery  * Copyright © 2018 Intel Corporation
3e7329c71SVernon Mauery  *
4e7329c71SVernon Mauery  * Licensed under the Apache License, Version 2.0 (the "License");
5e7329c71SVernon Mauery  * you may not use this file except in compliance with the License.
6e7329c71SVernon Mauery  * You may obtain a copy of the License at
7e7329c71SVernon Mauery  *
8e7329c71SVernon Mauery  *     http://www.apache.org/licenses/LICENSE-2.0
9e7329c71SVernon Mauery  *
10e7329c71SVernon Mauery  * Unless required by applicable law or agreed to in writing, software
11e7329c71SVernon Mauery  * distributed under the License is distributed on an "AS IS" BASIS,
12e7329c71SVernon Mauery  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13e7329c71SVernon Mauery  * See the License for the specific language governing permissions and
14e7329c71SVernon Mauery  * limitations under the License.
15e7329c71SVernon Mauery  */
16e7329c71SVernon Mauery #pragma once
17e7329c71SVernon Mauery 
18e7329c71SVernon Mauery #include <boost/asio/spawn.hpp>
19e08fbffcSVernon Mauery #include <ipmid/api-types.hpp>
20e7329c71SVernon Mauery #include <ipmid/message/types.hpp>
21997952afSVernon Mauery #include <ipmid/types.hpp>
22*b4b40918SGeorge Liu #include <phosphor-logging/lg2.hpp>
2333298af1SVernon Mauery #include <sdbusplus/asio/connection.hpp>
24fbc6c9d7SPatrick Williams 
25fbc6c9d7SPatrick Williams #include <algorithm>
26fbc6c9d7SPatrick Williams #include <cstdint>
27fbc6c9d7SPatrick Williams #include <exception>
28fbc6c9d7SPatrick Williams #include <memory>
29e7329c71SVernon Mauery #include <tuple>
30e7329c71SVernon Mauery #include <utility>
31e7329c71SVernon Mauery #include <vector>
32e7329c71SVernon Mauery 
33e7329c71SVernon Mauery namespace ipmi
34e7329c71SVernon Mauery {
35e7329c71SVernon Mauery 
36e7329c71SVernon Mauery struct Context
37e7329c71SVernon Mauery {
38e7329c71SVernon Mauery     using ptr = std::shared_ptr<Context>;
39e7329c71SVernon Mauery 
4033298af1SVernon Mauery     Context() = delete;
4133298af1SVernon Mauery     Context(const Context&) = default;
4233298af1SVernon Mauery     Context& operator=(const Context&) = default;
4333298af1SVernon Mauery     Context(Context&&) = delete;
4433298af1SVernon Mauery     Context& operator=(Context&&) = delete;
45e7329c71SVernon Mauery 
Contextipmi::Context4633298af1SVernon Mauery     Context(std::shared_ptr<sdbusplus::asio::connection> bus, NetFn netFn,
47c11cc5c0SJohnathan Mantey             uint8_t lun, Cmd cmd, int channel, int userId, uint32_t sessionId,
48f7d081f6SKumar Thangavel             Privilege priv, int rqSA, int hostIdx,
49f7d081f6SKumar Thangavel             boost::asio::yield_context& yield) :
501318a5edSPatrick Williams         bus(bus), netFn(netFn), lun(lun), cmd(cmd), channel(channel),
511318a5edSPatrick Williams         userId(userId), sessionId(sessionId), priv(priv), rqSA(rqSA),
521318a5edSPatrick Williams         hostIdx(hostIdx), yield(yield)
53fbc6c9d7SPatrick Williams     {}
54e7329c71SVernon Mauery 
5533298af1SVernon Mauery     std::shared_ptr<sdbusplus::asio::connection> bus;
56e7329c71SVernon Mauery     // normal IPMI context (what call is this, from whence it came...)
5733298af1SVernon Mauery     NetFn netFn;
58c11cc5c0SJohnathan Mantey     uint8_t lun;
5933298af1SVernon Mauery     Cmd cmd;
6033298af1SVernon Mauery     int channel;
6133298af1SVernon Mauery     int userId;
624d22640aSRajashekar Gade Reddy     uint32_t sessionId;
6333298af1SVernon Mauery     Privilege priv;
64d6a2da07SVernon Mauery     // srcAddr is only set on IPMB requests because
65d6a2da07SVernon Mauery     // Platform Event Message needs it to determine the incoming format
6633298af1SVernon Mauery     int rqSA;
67f7d081f6SKumar Thangavel     int hostIdx;
68cb09aa00SJames Feist     boost::asio::yield_context yield;
69e7329c71SVernon Mauery };
70e7329c71SVernon Mauery 
71e7329c71SVernon Mauery namespace message
72e7329c71SVernon Mauery {
73e7329c71SVernon Mauery 
74e7329c71SVernon Mauery namespace details
75e7329c71SVernon Mauery {
76e7329c71SVernon Mauery 
77e7329c71SVernon Mauery template <typename A>
78e7329c71SVernon Mauery struct UnpackSingle;
79e7329c71SVernon Mauery 
80e7329c71SVernon Mauery template <typename T>
81e7329c71SVernon Mauery using UnpackSingle_t = UnpackSingle<utility::TypeIdDowncast_t<T>>;
82e7329c71SVernon Mauery 
83e7329c71SVernon Mauery template <typename A>
84e7329c71SVernon Mauery struct PackSingle;
85e7329c71SVernon Mauery 
86e7329c71SVernon Mauery template <typename T>
87e7329c71SVernon Mauery using PackSingle_t = PackSingle<utility::TypeIdDowncast_t<T>>;
88e7329c71SVernon Mauery 
89e7329c71SVernon Mauery // size to hold 64 bits plus one (possibly-)partial byte
90e7329c71SVernon Mauery static constexpr size_t bitStreamSize = ((sizeof(uint64_t) + 1) * CHAR_BIT);
91e7329c71SVernon Mauery 
92e7329c71SVernon Mauery } // namespace details
93e7329c71SVernon Mauery 
94e7329c71SVernon Mauery /**
95e7329c71SVernon Mauery  * @brief a payload class that provides a mechanism to pack and unpack data
96e7329c71SVernon Mauery  *
97e7329c71SVernon Mauery  * When a new request is being executed, the Payload class is responsible for
98e7329c71SVernon Mauery  * attempting to unpack all the required arguments from the incoming blob. For
99e7329c71SVernon Mauery  * variable-length functions, it is possible to have function signature have a
100e7329c71SVernon Mauery  * Payload object, which will then allow the remaining data to be extracted as
101e7329c71SVernon Mauery  * needed.
102e7329c71SVernon Mauery  *
103e7329c71SVernon Mauery  * When creating a response, the parameters returned from the callback use a
104e7329c71SVernon Mauery  * newly created payload object to pack all the parameters into a buffer that is
105e7329c71SVernon Mauery  * then returned to the requester.
106e7329c71SVernon Mauery  *
107e7329c71SVernon Mauery  * These interfaces make calls into the message/pack.hpp and message/unpack.hpp
108e7329c71SVernon Mauery  * functions.
109e7329c71SVernon Mauery  */
110e7329c71SVernon Mauery struct Payload
111e7329c71SVernon Mauery {
112e7329c71SVernon Mauery     Payload() = default;
113e7329c71SVernon Mauery     Payload(const Payload&) = default;
114e7329c71SVernon Mauery     Payload& operator=(const Payload&) = default;
115e7329c71SVernon Mauery     Payload(Payload&&) = default;
116e7329c71SVernon Mauery     Payload& operator=(Payload&&) = default;
117e7329c71SVernon Mauery 
Payloadipmi::message::Payload118fbc6c9d7SPatrick Williams     explicit Payload(SecureBuffer&& data) : raw(std::move(data)) {}
119e7329c71SVernon Mauery 
~Payloadipmi::message::Payload120e7329c71SVernon Mauery     ~Payload()
121e7329c71SVernon Mauery     {
122f2fd17a4SWilliam A. Kennington III         if (raw.size() != 0 && std::uncaught_exceptions() == 0 && !trailingOk &&
123f2fd17a4SWilliam A. Kennington III             !unpackCheck && !unpackError)
124e7329c71SVernon Mauery         {
125*b4b40918SGeorge Liu             lg2::error(
126*b4b40918SGeorge Liu                 "Failed to check request for full unpack: raw size: {RAW_SIZE}",
127*b4b40918SGeorge Liu                 "RAW_SIZE", raw.size());
128e7329c71SVernon Mauery         }
129e7329c71SVernon Mauery     }
130e7329c71SVernon Mauery 
131e7329c71SVernon Mauery     /******************************************************************
132e7329c71SVernon Mauery      * raw vector access
133e7329c71SVernon Mauery      *****************************************************************/
134e7329c71SVernon Mauery     /**
135e7329c71SVernon Mauery      * @brief return the size of the underlying raw buffer
136e7329c71SVernon Mauery      */
sizeipmi::message::Payload137e7329c71SVernon Mauery     size_t size() const
138e7329c71SVernon Mauery     {
139e7329c71SVernon Mauery         return raw.size();
140e7329c71SVernon Mauery     }
141e7329c71SVernon Mauery     /**
142e7329c71SVernon Mauery      * @brief resize the underlying raw buffer to a new size
143e7329c71SVernon Mauery      *
144e7329c71SVernon Mauery      * @param sz - new size for the buffer
145e7329c71SVernon Mauery      */
resizeipmi::message::Payload146e7329c71SVernon Mauery     void resize(size_t sz)
147e7329c71SVernon Mauery     {
148e7329c71SVernon Mauery         raw.resize(sz);
149e7329c71SVernon Mauery     }
150e7329c71SVernon Mauery     /**
151e7329c71SVernon Mauery      * @brief return a pointer to the underlying raw buffer
152e7329c71SVernon Mauery      */
dataipmi::message::Payload153e7329c71SVernon Mauery     uint8_t* data()
154e7329c71SVernon Mauery     {
155e7329c71SVernon Mauery         return raw.data();
156e7329c71SVernon Mauery     }
157e7329c71SVernon Mauery     /**
158e7329c71SVernon Mauery      * @brief return a const pointer to the underlying raw buffer
159e7329c71SVernon Mauery      */
dataipmi::message::Payload160e7329c71SVernon Mauery     const uint8_t* data() const
161e7329c71SVernon Mauery     {
162e7329c71SVernon Mauery         return raw.data();
163e7329c71SVernon Mauery     }
164e7329c71SVernon Mauery 
165e7329c71SVernon Mauery     /******************************************************************
166e7329c71SVernon Mauery      * Response operations
167e7329c71SVernon Mauery      *****************************************************************/
168e7329c71SVernon Mauery     /**
169e7329c71SVernon Mauery      * @brief append a series of bytes to the buffer
170e7329c71SVernon Mauery      *
171e7329c71SVernon Mauery      * @tparam T - the type pointer to return; must be compatible to a byte
172e7329c71SVernon Mauery      *
173e7329c71SVernon Mauery      * @param begin - a pointer to the beginning of the series
174e7329c71SVernon Mauery      * @param end - a pointer to the end of the series
175e7329c71SVernon Mauery      */
176e7329c71SVernon Mauery     template <typename T>
appendipmi::message::Payload177e7329c71SVernon Mauery     void append(T* begin, T* end)
178e7329c71SVernon Mauery     {
179e7329c71SVernon Mauery         static_assert(
180e7329c71SVernon Mauery             std::is_same_v<utility::TypeIdDowncast_t<T>, int8_t> ||
181e7329c71SVernon Mauery                 std::is_same_v<utility::TypeIdDowncast_t<T>, uint8_t> ||
182e7329c71SVernon Mauery                 std::is_same_v<utility::TypeIdDowncast_t<T>, char>,
183e7329c71SVernon Mauery             "begin and end must be signed or unsigned byte pointers");
184e7329c71SVernon Mauery         // this interface only allows full-byte access; pack in partial bytes
185e7329c71SVernon Mauery         drain();
186e7329c71SVernon Mauery         raw.insert(raw.end(), reinterpret_cast<const uint8_t*>(begin),
187e7329c71SVernon Mauery                    reinterpret_cast<const uint8_t*>(end));
188e7329c71SVernon Mauery     }
189e7329c71SVernon Mauery 
190e7329c71SVernon Mauery     /**
191e7329c71SVernon Mauery      * @brief append a series of bits to the buffer
192e7329c71SVernon Mauery      *
193e7329c71SVernon Mauery      * Only the lowest @count order of bits will be appended, with the most
194e7329c71SVernon Mauery      * significant of those bits getting appended first.
195e7329c71SVernon Mauery      *
196e7329c71SVernon Mauery      * @param count - number of bits to append
197e7329c71SVernon Mauery      * @param bits - a byte with count significant bits to append
198e7329c71SVernon Mauery      */
appendBitsipmi::message::Payload199e7329c71SVernon Mauery     void appendBits(size_t count, uint8_t bits)
200e7329c71SVernon Mauery     {
201e7329c71SVernon Mauery         // drain whole bytes out
202e7329c71SVernon Mauery         drain(true);
203e7329c71SVernon Mauery 
204e7329c71SVernon Mauery         // add in the new bits as the higher-order bits, filling LSBit first
205e7329c71SVernon Mauery         fixed_uint_t<details::bitStreamSize> tmp = bits;
206e7329c71SVernon Mauery         tmp <<= bitCount;
207e7329c71SVernon Mauery         bitStream |= tmp;
208e7329c71SVernon Mauery         bitCount += count;
209e7329c71SVernon Mauery 
210e7329c71SVernon Mauery         // drain any whole bytes we have appended
211e7329c71SVernon Mauery         drain(true);
212e7329c71SVernon Mauery     }
213e7329c71SVernon Mauery 
214e7329c71SVernon Mauery     /**
215e7329c71SVernon Mauery      * @brief empty out the bucket and pack it as bytes LSB-first
216e7329c71SVernon Mauery      *
217e7329c71SVernon Mauery      * @param wholeBytesOnly - if true, only the whole bytes will be drained
218e7329c71SVernon Mauery      */
drainipmi::message::Payload219e7329c71SVernon Mauery     void drain(bool wholeBytesOnly = false)
220e7329c71SVernon Mauery     {
221e7329c71SVernon Mauery         while (bitCount > 0)
222e7329c71SVernon Mauery         {
223e7329c71SVernon Mauery             uint8_t retVal;
224e7329c71SVernon Mauery             if (bitCount < CHAR_BIT)
225e7329c71SVernon Mauery             {
226e7329c71SVernon Mauery                 if (wholeBytesOnly)
227e7329c71SVernon Mauery                 {
228e7329c71SVernon Mauery                     break;
229e7329c71SVernon Mauery                 }
230e7329c71SVernon Mauery             }
231e7329c71SVernon Mauery             size_t bitsOut = std::min(static_cast<size_t>(CHAR_BIT), bitCount);
232e7329c71SVernon Mauery             retVal = static_cast<uint8_t>(bitStream);
233e7329c71SVernon Mauery             raw.push_back(retVal);
234e7329c71SVernon Mauery             bitStream >>= bitsOut;
235e7329c71SVernon Mauery             bitCount -= bitsOut;
236e7329c71SVernon Mauery         }
237e7329c71SVernon Mauery     }
238e7329c71SVernon Mauery 
239e7329c71SVernon Mauery     // base empty pack
packipmi::message::Payload240e7329c71SVernon Mauery     int pack()
241e7329c71SVernon Mauery     {
242e7329c71SVernon Mauery         return 0;
243e7329c71SVernon Mauery     }
244e7329c71SVernon Mauery 
245e7329c71SVernon Mauery     /**
246e7329c71SVernon Mauery      * @brief pack arbitrary values (of any supported type) into the buffer
247e7329c71SVernon Mauery      *
248e7329c71SVernon Mauery      * @tparam Arg - the type of the first argument
249e7329c71SVernon Mauery      * @tparam Args - the type of the optional remaining arguments
250e7329c71SVernon Mauery      *
251e7329c71SVernon Mauery      * @param arg - the first argument to pack
252e7329c71SVernon Mauery      * @param args... - the optional remaining arguments to pack
253e7329c71SVernon Mauery      *
254e7329c71SVernon Mauery      * @return int - non-zero on pack errors
255e7329c71SVernon Mauery      */
256e7329c71SVernon Mauery     template <typename Arg, typename... Args>
packipmi::message::Payload257e7329c71SVernon Mauery     int pack(Arg&& arg, Args&&... args)
258e7329c71SVernon Mauery     {
2591318a5edSPatrick Williams         int packRet =
2601318a5edSPatrick Williams             details::PackSingle_t<Arg>::op(*this, std::forward<Arg>(arg));
261e7329c71SVernon Mauery         if (packRet)
262e7329c71SVernon Mauery         {
263e7329c71SVernon Mauery             return packRet;
264e7329c71SVernon Mauery         }
265e7329c71SVernon Mauery         packRet = pack(std::forward<Args>(args)...);
266e7329c71SVernon Mauery         drain();
267e7329c71SVernon Mauery         return packRet;
268e7329c71SVernon Mauery     }
269e7329c71SVernon Mauery 
27092476a84SWilliam A. Kennington III     /**
27192476a84SWilliam A. Kennington III      * @brief Prepends another payload to this one
27292476a84SWilliam A. Kennington III      *
27392476a84SWilliam A. Kennington III      * Avoid using this unless absolutely required since it inserts into the
27492476a84SWilliam A. Kennington III      * front of the response payload.
27592476a84SWilliam A. Kennington III      *
27692476a84SWilliam A. Kennington III      * @param p - The payload to prepend
27792476a84SWilliam A. Kennington III      *
27892476a84SWilliam A. Kennington III      * @retunr int - non-zero on prepend errors
27992476a84SWilliam A. Kennington III      */
prependipmi::message::Payload28092476a84SWilliam A. Kennington III     int prepend(const ipmi::message::Payload& p)
28192476a84SWilliam A. Kennington III     {
28292476a84SWilliam A. Kennington III         if (bitCount != 0 || p.bitCount != 0)
28392476a84SWilliam A. Kennington III         {
28492476a84SWilliam A. Kennington III             return 1;
28592476a84SWilliam A. Kennington III         }
28692476a84SWilliam A. Kennington III         raw.reserve(raw.size() + p.raw.size());
28792476a84SWilliam A. Kennington III         raw.insert(raw.begin(), p.raw.begin(), p.raw.end());
28892476a84SWilliam A. Kennington III         return 0;
28992476a84SWilliam A. Kennington III     }
29092476a84SWilliam A. Kennington III 
291e7329c71SVernon Mauery     /******************************************************************
292e7329c71SVernon Mauery      * Request operations
293e7329c71SVernon Mauery      *****************************************************************/
294e7329c71SVernon Mauery     /**
295e7329c71SVernon Mauery      * @brief pop a series of bytes from the raw buffer
296e7329c71SVernon Mauery      *
297e7329c71SVernon Mauery      * @tparam T - the type pointer to return; must be compatible to a byte
298e7329c71SVernon Mauery      *
299e7329c71SVernon Mauery      * @param count - the number of bytes to return
300e7329c71SVernon Mauery      *
301e7329c71SVernon Mauery      * @return - a tuple of pointers (begin,begin+count)
302e7329c71SVernon Mauery      */
303e7329c71SVernon Mauery     template <typename T>
popipmi::message::Payload304e7329c71SVernon Mauery     auto pop(size_t count)
305e7329c71SVernon Mauery     {
306e7329c71SVernon Mauery         static_assert(
307e7329c71SVernon Mauery             std::is_same_v<utility::TypeIdDowncast_t<T>, int8_t> ||
308e7329c71SVernon Mauery                 std::is_same_v<utility::TypeIdDowncast_t<T>, uint8_t> ||
309e7329c71SVernon Mauery                 std::is_same_v<utility::TypeIdDowncast_t<T>, char>,
310e7329c71SVernon Mauery             "T* must be signed or unsigned byte pointers");
311e7329c71SVernon Mauery         // this interface only allows full-byte access; skip partial bits
312e7329c71SVernon Mauery         if (bitCount)
313e7329c71SVernon Mauery         {
314e7329c71SVernon Mauery             // WARN on unused bits?
315e7329c71SVernon Mauery             discardBits();
316e7329c71SVernon Mauery         }
317e7329c71SVernon Mauery         if (count <= (raw.size() - rawIndex))
318e7329c71SVernon Mauery         {
319e7329c71SVernon Mauery             auto range = std::make_tuple(
320e7329c71SVernon Mauery                 reinterpret_cast<T*>(raw.data() + rawIndex),
321e7329c71SVernon Mauery                 reinterpret_cast<T*>(raw.data() + rawIndex + count));
322e7329c71SVernon Mauery             rawIndex += count;
323e7329c71SVernon Mauery             return range;
324e7329c71SVernon Mauery         }
325e7329c71SVernon Mauery         unpackError = true;
326e7329c71SVernon Mauery         return std::make_tuple(reinterpret_cast<T*>(NULL),
327e7329c71SVernon Mauery                                reinterpret_cast<T*>(NULL));
328e7329c71SVernon Mauery     }
329e7329c71SVernon Mauery 
330e7329c71SVernon Mauery     /**
331e7329c71SVernon Mauery      * @brief fill bit stream with at least count bits for consumption
332e7329c71SVernon Mauery      *
333e7329c71SVernon Mauery      * @param count - number of bit needed
334e7329c71SVernon Mauery      *
335e7329c71SVernon Mauery      * @return - unpackError
336e7329c71SVernon Mauery      */
fillBitsipmi::message::Payload337e7329c71SVernon Mauery     bool fillBits(size_t count)
338e7329c71SVernon Mauery     {
339e7329c71SVernon Mauery         // add more bits to the top end of the bitstream
340e7329c71SVernon Mauery         // so we consume bits least-significant first
341e7329c71SVernon Mauery         if (count > (details::bitStreamSize - CHAR_BIT))
342e7329c71SVernon Mauery         {
343e7329c71SVernon Mauery             unpackError = true;
344e7329c71SVernon Mauery             return unpackError;
345e7329c71SVernon Mauery         }
346e7329c71SVernon Mauery         while (bitCount < count)
347e7329c71SVernon Mauery         {
348e7329c71SVernon Mauery             if (rawIndex < raw.size())
349e7329c71SVernon Mauery             {
350e7329c71SVernon Mauery                 fixed_uint_t<details::bitStreamSize> tmp = raw[rawIndex++];
351e7329c71SVernon Mauery                 tmp <<= bitCount;
352e7329c71SVernon Mauery                 bitStream |= tmp;
353e7329c71SVernon Mauery                 bitCount += CHAR_BIT;
354e7329c71SVernon Mauery             }
355e7329c71SVernon Mauery             else
356e7329c71SVernon Mauery             {
357e7329c71SVernon Mauery                 // raw has run out of bytes to pop
358e7329c71SVernon Mauery                 unpackError = true;
359e7329c71SVernon Mauery                 return unpackError;
360e7329c71SVernon Mauery             }
361e7329c71SVernon Mauery         }
362e7329c71SVernon Mauery         return false;
363e7329c71SVernon Mauery     }
364e7329c71SVernon Mauery 
365e7329c71SVernon Mauery     /**
366e7329c71SVernon Mauery      * @brief consume count bits from bitstream (must call fillBits first)
367e7329c71SVernon Mauery      *
368e7329c71SVernon Mauery      * @param count - number of bit needed
369e7329c71SVernon Mauery      *
370e7329c71SVernon Mauery      * @return - count bits from stream
371e7329c71SVernon Mauery      */
popBitsipmi::message::Payload372e7329c71SVernon Mauery     uint8_t popBits(size_t count)
373e7329c71SVernon Mauery     {
374e7329c71SVernon Mauery         if (bitCount < count)
375e7329c71SVernon Mauery         {
376e7329c71SVernon Mauery             unpackError = true;
377e7329c71SVernon Mauery             return 0;
378e7329c71SVernon Mauery         }
379e7329c71SVernon Mauery         // consume bits low-order bits first
380e7329c71SVernon Mauery         auto bits = bitStream.convert_to<uint8_t>();
381e7329c71SVernon Mauery         bits &= ((1 << count) - 1);
382e7329c71SVernon Mauery         bitStream >>= count;
383e7329c71SVernon Mauery         bitCount -= count;
384e7329c71SVernon Mauery         return bits;
385e7329c71SVernon Mauery     }
386e7329c71SVernon Mauery 
387e7329c71SVernon Mauery     /**
388e7329c71SVernon Mauery      * @brief discard all partial bits
389e7329c71SVernon Mauery      */
discardBitsipmi::message::Payload390e7329c71SVernon Mauery     void discardBits()
391e7329c71SVernon Mauery     {
392e7329c71SVernon Mauery         bitStream = 0;
393e7329c71SVernon Mauery         bitCount = 0;
394e7329c71SVernon Mauery     }
395e7329c71SVernon Mauery 
396e7329c71SVernon Mauery     /**
397e7329c71SVernon Mauery      * @brief fully reset the unpack stream
398e7329c71SVernon Mauery      */
resetipmi::message::Payload399e7329c71SVernon Mauery     void reset()
400e7329c71SVernon Mauery     {
401e7329c71SVernon Mauery         discardBits();
402e7329c71SVernon Mauery         rawIndex = 0;
403e7329c71SVernon Mauery         unpackError = false;
404e7329c71SVernon Mauery     }
405e7329c71SVernon Mauery 
406e7329c71SVernon Mauery     /**
407e7329c71SVernon Mauery      * @brief check to see if the stream has been fully unpacked
408e7329c71SVernon Mauery      *
409e7329c71SVernon Mauery      * @return bool - true if the stream has been unpacked and has no errors
410e7329c71SVernon Mauery      */
fullyUnpackedipmi::message::Payload411e7329c71SVernon Mauery     bool fullyUnpacked()
412e7329c71SVernon Mauery     {
413e7329c71SVernon Mauery         unpackCheck = true;
414e7329c71SVernon Mauery         return raw.size() == rawIndex && bitCount == 0 && !unpackError;
415e7329c71SVernon Mauery     }
416e7329c71SVernon Mauery 
417e7329c71SVernon Mauery     // base empty unpack
unpackipmi::message::Payload418e7329c71SVernon Mauery     int unpack()
419e7329c71SVernon Mauery     {
420e7329c71SVernon Mauery         return 0;
421e7329c71SVernon Mauery     }
422e7329c71SVernon Mauery 
423e7329c71SVernon Mauery     /**
424e7329c71SVernon Mauery      * @brief unpack arbitrary values (of any supported type) from the buffer
425e7329c71SVernon Mauery      *
426e7329c71SVernon Mauery      * @tparam Arg - the type of the first argument
427e7329c71SVernon Mauery      * @tparam Args - the type of the optional remaining arguments
428e7329c71SVernon Mauery      *
429e7329c71SVernon Mauery      * @param arg - the first argument to unpack
430e7329c71SVernon Mauery      * @param args... - the optional remaining arguments to unpack
431e7329c71SVernon Mauery      *
432e7329c71SVernon Mauery      * @return int - non-zero for unpack error
433e7329c71SVernon Mauery      */
434e7329c71SVernon Mauery     template <typename Arg, typename... Args>
unpackipmi::message::Payload435e7329c71SVernon Mauery     int unpack(Arg&& arg, Args&&... args)
436e7329c71SVernon Mauery     {
437e7329c71SVernon Mauery         int unpackRet =
438e7329c71SVernon Mauery             details::UnpackSingle_t<Arg>::op(*this, std::forward<Arg>(arg));
439e7329c71SVernon Mauery         if (unpackRet)
440e7329c71SVernon Mauery         {
441e7329c71SVernon Mauery             unpackError = true;
442e7329c71SVernon Mauery             return unpackRet;
443e7329c71SVernon Mauery         }
444e7329c71SVernon Mauery         return unpack(std::forward<Args>(args)...);
445e7329c71SVernon Mauery     }
446e7329c71SVernon Mauery 
447e7329c71SVernon Mauery     /**
448e7329c71SVernon Mauery      * @brief unpack a tuple of values (of any supported type) from the buffer
449e7329c71SVernon Mauery      *
450e7329c71SVernon Mauery      * This will unpack the elements of the tuple as if each one was passed in
451e7329c71SVernon Mauery      * individually, as if passed into the above variadic function.
452e7329c71SVernon Mauery      *
453e7329c71SVernon Mauery      * @tparam Types - the implicitly declared list of the tuple element types
454e7329c71SVernon Mauery      *
455e7329c71SVernon Mauery      * @param t - the tuple of values to unpack
456e7329c71SVernon Mauery      *
457e7329c71SVernon Mauery      * @return int - non-zero on unpack error
458e7329c71SVernon Mauery      */
459e7329c71SVernon Mauery     template <typename... Types>
unpackipmi::message::Payload460e7329c71SVernon Mauery     int unpack(std::tuple<Types...>& t)
461e7329c71SVernon Mauery     {
462e7329c71SVernon Mauery         // roll back checkpoint so that unpacking a tuple is atomic
463e7329c71SVernon Mauery         size_t priorBitCount = bitCount;
464e7329c71SVernon Mauery         size_t priorIndex = rawIndex;
465e7329c71SVernon Mauery         fixed_uint_t<details::bitStreamSize> priorBits = bitStream;
466e7329c71SVernon Mauery 
4671318a5edSPatrick Williams         int ret =
4681318a5edSPatrick Williams             std::apply([this](Types&... args) { return unpack(args...); }, t);
469e7329c71SVernon Mauery         if (ret)
470e7329c71SVernon Mauery         {
471e7329c71SVernon Mauery             bitCount = priorBitCount;
472e7329c71SVernon Mauery             bitStream = priorBits;
473e7329c71SVernon Mauery             rawIndex = priorIndex;
474e7329c71SVernon Mauery         }
475e7329c71SVernon Mauery 
476e7329c71SVernon Mauery         return ret;
477e7329c71SVernon Mauery     }
478e7329c71SVernon Mauery 
479e7329c71SVernon Mauery     // partial bytes in the form of bits
480e7329c71SVernon Mauery     fixed_uint_t<details::bitStreamSize> bitStream;
481e7329c71SVernon Mauery     size_t bitCount = 0;
482997952afSVernon Mauery     SecureBuffer raw;
483e7329c71SVernon Mauery     size_t rawIndex = 0;
48451694c22SWilliam A. Kennington III     bool trailingOk = true;
48551694c22SWilliam A. Kennington III     bool unpackCheck = false;
486e7329c71SVernon Mauery     bool unpackError = false;
487e7329c71SVernon Mauery };
488e7329c71SVernon Mauery 
489e7329c71SVernon Mauery /**
490e7329c71SVernon Mauery  * @brief high-level interface to an IPMI response
491e7329c71SVernon Mauery  *
492e7329c71SVernon Mauery  * Make it easy to just pack in the response args from the callback into a
493e7329c71SVernon Mauery  * buffer that goes back to the requester.
494e7329c71SVernon Mauery  */
495e7329c71SVernon Mauery struct Response
496e7329c71SVernon Mauery {
497e7329c71SVernon Mauery     /* Define all of the basic class operations:
498e7329c71SVernon Mauery      *     Not allowed:
499e7329c71SVernon Mauery      *         - Default constructor to avoid nullptrs.
500e7329c71SVernon Mauery      *     Allowed:
501e7329c71SVernon Mauery      *         - Copy operations.
502e7329c71SVernon Mauery      *         - Move operations.
503e7329c71SVernon Mauery      *         - Destructor.
504e7329c71SVernon Mauery      */
505e7329c71SVernon Mauery     Response() = delete;
506e7329c71SVernon Mauery     Response(const Response&) = default;
507e7329c71SVernon Mauery     Response& operator=(const Response&) = default;
508e7329c71SVernon Mauery     Response(Response&&) = default;
509e7329c71SVernon Mauery     Response& operator=(Response&&) = default;
510e7329c71SVernon Mauery     ~Response() = default;
511e7329c71SVernon Mauery 
512e7329c71SVernon Mauery     using ptr = std::shared_ptr<Response>;
513e7329c71SVernon Mauery 
Responseipmi::message::Response514e7329c71SVernon Mauery     explicit Response(Context::ptr& context) :
515e7329c71SVernon Mauery         payload(), ctx(context), cc(ccSuccess)
516fbc6c9d7SPatrick Williams     {}
517e7329c71SVernon Mauery 
518e7329c71SVernon Mauery     /**
519e7329c71SVernon Mauery      * @brief pack arbitrary values (of any supported type) into the payload
520e7329c71SVernon Mauery      *
521e7329c71SVernon Mauery      * @tparam Args - the type of the optional arguments
522e7329c71SVernon Mauery      *
523e7329c71SVernon Mauery      * @param args... - the optional arguments to pack
524e7329c71SVernon Mauery      *
525e7329c71SVernon Mauery      * @return int - non-zero on pack errors
526e7329c71SVernon Mauery      */
527e7329c71SVernon Mauery     template <typename... Args>
packipmi::message::Response528e7329c71SVernon Mauery     int pack(Args&&... args)
529e7329c71SVernon Mauery     {
530e7329c71SVernon Mauery         return payload.pack(std::forward<Args>(args)...);
531e7329c71SVernon Mauery     }
532e7329c71SVernon Mauery 
533e7329c71SVernon Mauery     /**
534e7329c71SVernon Mauery      * @brief pack a tuple of values (of any supported type) into the payload
535e7329c71SVernon Mauery      *
536e7329c71SVernon Mauery      * This will pack the elements of the tuple as if each one was passed in
537e7329c71SVernon Mauery      * individually, as if passed into the above variadic function.
538e7329c71SVernon Mauery      *
539e7329c71SVernon Mauery      * @tparam Types - the implicitly declared list of the tuple element types
540e7329c71SVernon Mauery      *
541e7329c71SVernon Mauery      * @param t - the tuple of values to pack
542e7329c71SVernon Mauery      *
543e7329c71SVernon Mauery      * @return int - non-zero on pack errors
544e7329c71SVernon Mauery      */
545e7329c71SVernon Mauery     template <typename... Types>
packipmi::message::Response546e7329c71SVernon Mauery     int pack(std::tuple<Types...>& t)
547e7329c71SVernon Mauery     {
548e7329c71SVernon Mauery         return payload.pack(t);
549e7329c71SVernon Mauery     }
550e7329c71SVernon Mauery 
55192476a84SWilliam A. Kennington III     /**
55292476a84SWilliam A. Kennington III      * @brief Prepends another payload to this one
55392476a84SWilliam A. Kennington III      *
55492476a84SWilliam A. Kennington III      * Avoid using this unless absolutely required since it inserts into the
55592476a84SWilliam A. Kennington III      * front of the response payload.
55692476a84SWilliam A. Kennington III      *
55792476a84SWilliam A. Kennington III      * @param p - The payload to prepend
55892476a84SWilliam A. Kennington III      *
55992476a84SWilliam A. Kennington III      * @retunr int - non-zero on prepend errors
56092476a84SWilliam A. Kennington III      */
prependipmi::message::Response56192476a84SWilliam A. Kennington III     int prepend(const ipmi::message::Payload& p)
56292476a84SWilliam A. Kennington III     {
56392476a84SWilliam A. Kennington III         return payload.prepend(p);
56492476a84SWilliam A. Kennington III     }
56592476a84SWilliam A. Kennington III 
566e7329c71SVernon Mauery     Payload payload;
567e7329c71SVernon Mauery     Context::ptr ctx;
568e7329c71SVernon Mauery     Cc cc;
569e7329c71SVernon Mauery };
570e7329c71SVernon Mauery 
571e7329c71SVernon Mauery /**
572e7329c71SVernon Mauery  * @brief high-level interface to an IPMI request
573e7329c71SVernon Mauery  *
574e7329c71SVernon Mauery  * Make it easy to unpack the buffer into the request args for the callback.
575e7329c71SVernon Mauery  */
576e7329c71SVernon Mauery struct Request
577e7329c71SVernon Mauery {
578e7329c71SVernon Mauery     /* Define all of the basic class operations:
579e7329c71SVernon Mauery      *     Not allowed:
580e7329c71SVernon Mauery      *         - Default constructor to avoid nullptrs.
581e7329c71SVernon Mauery      *     Allowed:
582e7329c71SVernon Mauery      *         - Copy operations.
583e7329c71SVernon Mauery      *         - Move operations.
584e7329c71SVernon Mauery      *         - Destructor.
585e7329c71SVernon Mauery      */
586e7329c71SVernon Mauery     Request() = delete;
587e7329c71SVernon Mauery     Request(const Request&) = default;
588e7329c71SVernon Mauery     Request& operator=(const Request&) = default;
589e7329c71SVernon Mauery     Request(Request&&) = default;
590e7329c71SVernon Mauery     Request& operator=(Request&&) = default;
591e7329c71SVernon Mauery     ~Request() = default;
592e7329c71SVernon Mauery 
593e7329c71SVernon Mauery     using ptr = std::shared_ptr<Request>;
594e7329c71SVernon Mauery 
Requestipmi::message::Request595997952afSVernon Mauery     explicit Request(Context::ptr context, SecureBuffer&& d) :
596997952afSVernon Mauery         payload(std::forward<SecureBuffer>(d)), ctx(context)
597fbc6c9d7SPatrick Williams     {}
598e7329c71SVernon Mauery 
599e7329c71SVernon Mauery     /**
600e7329c71SVernon Mauery      * @brief unpack arbitrary values (of any supported type) from the payload
601e7329c71SVernon Mauery      *
602e7329c71SVernon Mauery      * @tparam Args - the type of the optional arguments
603e7329c71SVernon Mauery      *
604e7329c71SVernon Mauery      * @param args... - the optional arguments to unpack
605e7329c71SVernon Mauery      *
606e7329c71SVernon Mauery      * @return int - non-zero for unpack error
607e7329c71SVernon Mauery      */
608e7329c71SVernon Mauery     template <typename... Args>
unpackipmi::message::Request609e7329c71SVernon Mauery     int unpack(Args&&... args)
610e7329c71SVernon Mauery     {
611e7329c71SVernon Mauery         int unpackRet = payload.unpack(std::forward<Args>(args)...);
612f865dea9SVernon Mauery         if (unpackRet != ipmi::ccSuccess)
613e7329c71SVernon Mauery         {
614f865dea9SVernon Mauery             // not all bits were consumed by requested parameters
615f865dea9SVernon Mauery             return ipmi::ccReqDataLenInvalid;
616f865dea9SVernon Mauery         }
617e7329c71SVernon Mauery         if (!payload.trailingOk)
618e7329c71SVernon Mauery         {
619e7329c71SVernon Mauery             if (!payload.fullyUnpacked())
620e7329c71SVernon Mauery             {
621e7329c71SVernon Mauery                 // not all bits were consumed by requested parameters
622e7329c71SVernon Mauery                 return ipmi::ccReqDataLenInvalid;
623e7329c71SVernon Mauery             }
624e7329c71SVernon Mauery         }
625f865dea9SVernon Mauery         return ipmi::ccSuccess;
626e7329c71SVernon Mauery     }
627e7329c71SVernon Mauery 
628e7329c71SVernon Mauery     /**
629e7329c71SVernon Mauery      * @brief unpack a tuple of values (of any supported type) from the payload
630e7329c71SVernon Mauery      *
631e7329c71SVernon Mauery      * This will unpack the elements of the tuple as if each one was passed in
632e7329c71SVernon Mauery      * individually, as if passed into the above variadic function.
633e7329c71SVernon Mauery      *
634e7329c71SVernon Mauery      * @tparam Types - the implicitly declared list of the tuple element types
635e7329c71SVernon Mauery      *
636e7329c71SVernon Mauery      * @param t - the tuple of values to unpack
637e7329c71SVernon Mauery      *
638e7329c71SVernon Mauery      * @return int - non-zero on unpack error
639e7329c71SVernon Mauery      */
640e7329c71SVernon Mauery     template <typename... Types>
unpackipmi::message::Request641e7329c71SVernon Mauery     int unpack(std::tuple<Types...>& t)
642e7329c71SVernon Mauery     {
643e7329c71SVernon Mauery         return std::apply([this](Types&... args) { return unpack(args...); },
644e7329c71SVernon Mauery                           t);
645e7329c71SVernon Mauery     }
646e7329c71SVernon Mauery 
647e7329c71SVernon Mauery     /** @brief Create a response message that corresponds to this request
648e7329c71SVernon Mauery      *
649e7329c71SVernon Mauery      * @return A shared_ptr to the response message created
650e7329c71SVernon Mauery      */
makeResponseipmi::message::Request651e7329c71SVernon Mauery     Response::ptr makeResponse()
652e7329c71SVernon Mauery     {
653e7329c71SVernon Mauery         return std::make_shared<Response>(ctx);
654e7329c71SVernon Mauery     }
655e7329c71SVernon Mauery 
656e7329c71SVernon Mauery     Payload payload;
657e7329c71SVernon Mauery     Context::ptr ctx;
658e7329c71SVernon Mauery };
659e7329c71SVernon Mauery 
660e7329c71SVernon Mauery } // namespace message
661e7329c71SVernon Mauery 
662e7329c71SVernon Mauery } // namespace ipmi
663e7329c71SVernon Mauery 
664e7329c71SVernon Mauery // include packing and unpacking of types
665e7329c71SVernon Mauery #include <ipmid/message/pack.hpp>
666e7329c71SVernon Mauery #include <ipmid/message/unpack.hpp>
667