blob: 3467ccd7115903a769b0a6ebf396eb58aa52c7b6 [file] [log] [blame]
#pragma once
#include "config.h"
#include "data_handler.hpp"
#include "image_handler.hpp"
#include "util.hpp"
#include <blobs-ipmid/blobs.hpp>
#include <cstdint>
#include <map>
#include <memory>
#if HAVE_SDBUSPLUS
#include <sdbusplus/bus.hpp>
#else
namespace sdbusplus
{
namespace bus
{
class bus
{
};
} // namespace bus
} // namespace sdbusplus
#endif
#include <string>
#include <vector>
namespace blobs
{
/**
* Representation of a session, includes how to read/write data.
*/
struct Session
{
/**
* Built a session object.
*
* @param[in] the active path to which this corresponds.
*/
explicit Session(const std::string& path) :
dataHandler(nullptr), imageHandler(nullptr), flags(0),
state(State::closed), activePath(path)
{
}
/**
* Pointer to the correct Data handler interface. (nullptr on BT (or KCS))
*/
DataInterface* dataHandler;
/**
* Pointer to the correct image handler interface. (nullptr on hash
* blob_id)
*/
ImageHandlerInterface* imageHandler;
/** The flags used to open the session. */
std::uint16_t flags;
/** A sesion can be for an image (or tarball) or the hash. */
enum class State
{
open = 0,
closed = 1,
};
/** The current state of this session. */
State state;
/** The active path. */
std::string activePath;
};
struct ExtChunkHdr
{
std::uint32_t length; /* Length of the data queued (little endian). */
} __attribute__((packed));
/**
* Register only one firmware blob handler that will manage all sessions.
*/
class FirmwareBlobHandler : public GenericBlobInterface
{
public:
enum UpdateFlags : std::uint16_t
{
openRead = (1 << 0), /* Flag for reading. */
openWrite = (1 << 1), /* Flag for writing. */
ipmi = (1 << 8), /* Expect to send contents over IPMI BlockTransfer. */
p2a = (1 << 9), /* Expect to send contents over P2A bridge. */
lpc = (1 << 10), /* Expect to send contents over LPC bridge. */
};
/* TODO: All of the states may not be required - if we add abort() commands
* appropriately.
*/
/** The state of the firmware update process. */
enum class UpdateState
{
/** The initial state. */
notYetStarted = 0,
/** The BMC is expecting to receive bytes. */
uploadInProgress,
/** The BMC is ready for verification or more bytes. */
verificationPending,
/** The verification process has started, no more writes allowed. */
verificationStarted,
/** The verification process has completed. */
verificationCompleted,
};
/** The return values for verification. */
enum class VerifyCheckResponses : std::uint8_t
{
running = 0,
success = 1,
failed = 2,
other = 3,
};
/**
* Create a FirmwareBlobHandler.
*
* @param[in] bus - an sdbusplus handler for a bus to use.
* @param[in] firmwares - list of firmware blob_ids to support.
* @param[in] transports - list of transports to support.
* @param[in[ verificationPath - path to check for verification output
*/
static std::unique_ptr<GenericBlobInterface> CreateFirmwareBlobHandler(
sdbusplus::bus::bus&& bus, const std::vector<HandlerPack>& firmwares,
const std::vector<DataHandlerPack>& transports,
const std::string& verificationPath);
/**
* Create a FirmwareBlobHandler.
*
* @param[in] bus - an sdbusplus handler for a bus to use
* @param[in] firmwares - list of firmware types and their handlers
* @param[in] blobs - list of blobs_ids to support
* @param[in] transports - list of transport types and their handlers
* @param[in] bitmask - bitmask of transports to support
*/
FirmwareBlobHandler(sdbusplus::bus::bus&& bus,
const std::vector<HandlerPack>& firmwares,
const std::vector<std::string>& blobs,
const std::vector<DataHandlerPack>& transports,
std::uint16_t bitmask,
const std::string& verificationPath) :
bus(std::move(bus)),
handlers(firmwares), blobIDs(blobs), transports(transports),
bitmask(bitmask), activeImage(activeImageBlobId),
activeHash(activeHashBlobId), verifyImage(verifyBlobId), lookup(),
state(UpdateState::notYetStarted), verificationPath(verificationPath)
{
}
~FirmwareBlobHandler() = default;
FirmwareBlobHandler(const FirmwareBlobHandler&) = delete;
FirmwareBlobHandler& operator=(const FirmwareBlobHandler&) = delete;
FirmwareBlobHandler(FirmwareBlobHandler&&) = default;
FirmwareBlobHandler& operator=(FirmwareBlobHandler&&) = default;
bool canHandleBlob(const std::string& path) override;
std::vector<std::string> getBlobIds() override;
bool deleteBlob(const std::string& path) override;
bool stat(const std::string& path, struct BlobMeta* meta) override;
bool open(uint16_t session, uint16_t flags,
const std::string& path) override;
std::vector<uint8_t> read(uint16_t session, uint32_t offset,
uint32_t requestedSize) override;
bool write(uint16_t session, uint32_t offset,
const std::vector<uint8_t>& data) override;
bool writeMeta(uint16_t session, uint32_t offset,
const std::vector<uint8_t>& data) override;
bool commit(uint16_t session, const std::vector<uint8_t>& data) override;
bool close(uint16_t session) override;
bool stat(uint16_t session, struct BlobMeta* meta) override;
bool expire(uint16_t session) override;
bool triggerVerification();
/** Allow grabbing the current state. */
UpdateState getCurrentState() const
{
return state;
};
private:
sdbusplus::bus::bus bus;
/** List of handlers by type. */
std::vector<HandlerPack> handlers;
/** Active list of blobIDs. */
std::vector<std::string> blobIDs;
/** List of handlers by transport type. */
std::vector<DataHandlerPack> transports;
/** The bits set indicate what transport mechanisms are supported. */
std::uint16_t bitmask;
/** Active image session. */
Session activeImage;
/** Active hash session. */
Session activeHash;
/** Session for verification. */
Session verifyImage;
/** A quick method for looking up a session's mechanisms and details. */
std::map<std::uint16_t, Session*> lookup;
/** The firmware update state. */
UpdateState state;
const std::string verificationPath;
/** Temporary variable to track whether a blob is open. */
bool fileOpen = false;
};
} // namespace blobs