blob: 1883128b55559558de37cec6a44ae881e778acf8 [file] [log] [blame]
// Copyright 2021 Google Inc.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
#include "log_handler.hpp"
#include <algorithm>
#include <cstring>
#include <ios>
#include <limits>
#include <memory>
#include <optional>
#include <utility>
#include <vector>
namespace ipmi_flash
{
LogBlobHandler::LogBlobHandler(std::vector<HandlerConfig<ActionPack>>&& configs)
{
for (auto& config : configs)
{
auto info = std::make_unique<BlobInfo>();
info->blobId = std::move(config.blobId);
info->actions = std::move(config.actions);
info->handler = std::move(config.handler);
info->actions->onOpen->setCallback(
[infoP = info.get()](TriggerableActionInterface& tai) {
auto data =
std::make_shared<std::optional<std::vector<uint8_t>>>();
do
{
if (tai.status() != ActionStatus::success)
{
fprintf(stderr,
"LogBlobHandler: Log file unit failed for %s\n",
infoP->blobId.c_str());
continue;
}
if (!infoP->handler->open("", std::ios::in))
{
fprintf(
stderr,
"LogBlobHandler: Opening log file failed for %s\n",
infoP->blobId.c_str());
continue;
}
auto d = infoP->handler->read(
0, std::numeric_limits<uint32_t>::max());
infoP->handler->close();
if (!d)
{
fprintf(
stderr,
"LogBlobHandler: Reading log file failed for %s\n",
infoP->blobId.c_str());
continue;
}
*data = std::move(d);
} while (false);
for (auto sessionP : infoP->sessionsToUpdate)
{
sessionP->data = data;
}
infoP->sessionsToUpdate.clear();
});
if (!blobInfoMap.try_emplace(info->blobId, std::move(info)).second)
{
fprintf(stderr,
"LogBlobHandler: Ignoring duplicate config for %s\n",
info->blobId.c_str());
}
}
}
bool LogBlobHandler::canHandleBlob(const std::string& path)
{
return blobInfoMap.find(path) != blobInfoMap.end();
}
std::vector<std::string> LogBlobHandler::getBlobIds()
{
std::vector<std::string> ret;
for (const auto& [key, _] : blobInfoMap)
{
ret.emplace_back(key);
}
return ret;
}
/**
* deleteBlob - does nothing, always fails
*/
bool LogBlobHandler::deleteBlob(const std::string& path)
{
for (const auto& [sessionId, sessionInfo] : sessionInfoMap)
{
if (sessionInfo->blob->blobId == path)
{
fprintf(stderr,
"LogBlobHandler: delete %s fail: there is an open session "
"for this blob\n",
path.c_str());
return false;
}
}
auto* blob = blobInfoMap.at(path).get();
if (!blob->actions->onDelete->trigger())
{
fprintf(stderr,
"LogBlobHandler: delete %s fail: onDelete trigger failed\n",
path.c_str());
return false;
}
return true;
}
bool LogBlobHandler::stat(const std::string&, blobs::BlobMeta*)
{
return false;
}
bool LogBlobHandler::open(uint16_t session, uint16_t flags,
const std::string& path)
{
/* only reads are supported, check if blob is handled and make sure
* the blob isn't already opened
*/
if (flags != blobs::read)
{
fprintf(stderr,
"LogBlobHandler: open %s fail: unsupported flags(0x%04X.)\n",
path.c_str(), flags);
return false;
}
auto info = std::make_unique<SessionInfo>();
info->blob = blobInfoMap.at(path).get();
info->blob->sessionsToUpdate.emplace(info.get());
if (info->blob->sessionsToUpdate.size() == 1 &&
!info->blob->actions->onOpen->trigger())
{
fprintf(stderr, "LogBlobHandler: open %s fail: onOpen trigger failed\n",
path.c_str());
info->blob->sessionsToUpdate.erase(info.get());
return false;
}
sessionInfoMap[session] = std::move(info);
return true;
}
std::vector<uint8_t> LogBlobHandler::read(uint16_t session, uint32_t offset,
uint32_t requestedSize)
{
auto& data = sessionInfoMap.at(session)->data;
if (data == nullptr || !*data)
{
throw std::runtime_error("LogBlobHandler: Log data not ready for read");
}
if ((*data)->size() < offset)
{
return {};
}
std::vector<uint8_t> ret(
std::min<size_t>(requestedSize, (*data)->size() - offset));
std::memcpy(&ret[0], &(**data)[offset], ret.size());
return ret;
}
bool LogBlobHandler::close(uint16_t session)
{
auto it = sessionInfoMap.find(session);
if (it == sessionInfoMap.end())
{
return false;
}
auto& info = *it->second;
info.blob->sessionsToUpdate.erase(&info);
if (info.blob->sessionsToUpdate.empty())
{
info.blob->actions->onOpen->abort();
}
sessionInfoMap.erase(it);
return true;
}
bool LogBlobHandler::stat(uint16_t session, blobs::BlobMeta* meta)
{
const auto& data = sessionInfoMap.at(session)->data;
if (data == nullptr)
{
meta->blobState = blobs::StateFlags::committing;
meta->size = 0;
}
else if (!*data)
{
meta->blobState = blobs::StateFlags::commit_error;
meta->size = 0;
}
else
{
meta->blobState =
blobs::StateFlags::committed | blobs::StateFlags::open_read;
meta->size = (*data)->size();
}
return true;
}
bool LogBlobHandler::expire(uint16_t session)
{
close(session);
return true;
}
} // namespace ipmi_flash