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