blob: dacca193b92fa4100e99e50a89320fc5e97a206a [file] [log] [blame]
/*
* Copyright 2018 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 "file_handler.hpp"
#include <cstdint>
#include <filesystem>
#include <ios>
#include <memory>
#include <string>
#include <vector>
namespace ipmi_flash
{
namespace fs = std::filesystem;
bool FileHandler::open(const std::string& path, std::ios_base::openmode mode)
{
/* force binary mode */
mode |= std::ios::binary;
this->path = path;
if (file.is_open())
{
/* This wasn't properly closed somehow.
* TODO: Throw an error or just reset the state?
*/
return false;
}
file.open(filename, mode);
if (!file.good()) /* on success goodbit is set */
{
/* TODO: Oh no! Care about this. */
return false;
}
/* We were able to open the file for staging.
* TODO: We'll need to do other stuff to eventually.
*/
return true;
}
void FileHandler::close()
{
if (file.is_open())
{
file.close();
}
return;
}
bool FileHandler::write(std::uint32_t offset,
const std::vector<std::uint8_t>& data)
{
if (!file.is_open())
{
return false;
}
/* We could track this, but if they write in a scattered method, this is
* easier.
*/
file.seekp(offset, std::ios_base::beg);
if (!file.good())
{
/* the documentation wasn't super clear on fail vs bad in these cases,
* so let's only be happy with goodness.
*/
return false;
}
file.write(reinterpret_cast<const char*>(data.data()), data.size());
if (!file.good())
{
return false;
}
return true;
}
std::optional<std::vector<uint8_t>> FileHandler::read(std::uint32_t offset,
std::uint32_t size)
{
if (!file.is_open())
{
return std::nullopt;
}
/* determine size of file */
file.seekg(0, std::ios_base::end);
uint32_t filesize = file.tellg();
uint32_t bytesToRead = size;
/* make sure to not read past the end of file */
if (offset + size > filesize)
{
bytesToRead = filesize - offset;
}
/* if no bytes can be read, fail */
if (0 == bytesToRead)
{
return std::nullopt;
}
/* seek to offset then read */
file.seekg(offset);
std::vector<uint8_t> fileData(bytesToRead);
file.read(reinterpret_cast<char*>(fileData.data()), bytesToRead);
/* if any sort of failure happened during all the seeks
* and reads then fail the entire operation
*/
if (!file.good())
{
return std::nullopt;
}
return fileData;
}
int FileHandler::getSize()
{
try
{
return static_cast<int>(fs::file_size(filename));
}
catch (const fs::filesystem_error& e)
{}
return 0;
}
} // namespace ipmi_flash