1
0
mirror of synced 2024-12-15 17:21:16 +01:00
ImHex/lib/libimhex/source/helpers/file.cpp

150 lines
3.8 KiB
C++
Raw Normal View History

#include <hex/helpers/file.hpp>
#include <unistd.h>
#include <cstring>
namespace hex::fs {
File::File(const std::fs::path &path, Mode mode) noexcept : m_path(path) {
#if defined(OS_WINDOWS)
if (mode == File::Mode::Read)
this->m_file = _wfopen(path.c_str(), L"rb");
else if (mode == File::Mode::Write)
this->m_file = _wfopen(path.c_str(), L"r+b");
if (mode == File::Mode::Create || (mode == File::Mode::Write && this->m_file == nullptr))
this->m_file = _wfopen(path.c_str(), L"w+b");
#else
if (mode == File::Mode::Read)
this->m_file = fopen64(path.string().c_str(), "rb");
else if (mode == File::Mode::Write)
this->m_file = fopen64(path.string().c_str(), "r+b");
if (mode == File::Mode::Create || (mode == File::Mode::Write && this->m_file == nullptr))
this->m_file = fopen64(path.string().c_str(), "w+b");
#endif
}
File::File() noexcept {
2021-09-11 14:41:18 +02:00
this->m_file = nullptr;
}
2021-09-12 20:26:39 +02:00
File::File(File &&other) noexcept {
this->m_file = other.m_file;
other.m_file = nullptr;
}
File::~File() {
this->close();
}
File &File::operator=(File &&other) noexcept {
this->m_file = other.m_file;
other.m_file = nullptr;
this->m_path = std::move(other.m_path);
return *this;
}
void File::seek(u64 offset) {
fseeko64(this->m_file, offset, SEEK_SET);
}
void File::close() {
if (isValid()) {
2022-03-01 00:03:28 +01:00
std::fclose(this->m_file);
this->m_file = nullptr;
}
}
2021-09-11 14:41:18 +02:00
size_t File::readBuffer(u8 *buffer, size_t size) {
if (!isValid()) return 0;
return fread(buffer, size, 1, this->m_file);
}
std::vector<u8> File::readBytes(size_t numBytes) {
if (!isValid()) return {};
2021-09-11 14:41:18 +02:00
auto size = numBytes ?: getSize();
if (size == 0) return {};
std::vector<u8> bytes(size);
auto bytesRead = fread(bytes.data(), 1, bytes.size(), this->m_file);
bytes.resize(bytesRead);
return bytes;
}
std::string File::readString(size_t numBytes) {
if (!isValid()) return {};
2021-09-11 14:41:18 +02:00
if (getSize() == 0) return {};
auto bytes = readBytes(numBytes);
if (bytes.empty())
return "";
const char *cString = reinterpret_cast<const char *>(bytes.data());
2022-06-27 15:08:22 +02:00
return { cString, std::min(bytes.size(), std::strlen(cString)) };
}
2021-09-11 14:41:18 +02:00
void File::write(const u8 *buffer, size_t size) {
if (!isValid()) return;
2022-03-01 00:03:28 +01:00
std::fwrite(buffer, size, 1, this->m_file);
2021-09-11 14:41:18 +02:00
}
void File::write(const std::vector<u8> &bytes) {
2021-09-11 14:41:18 +02:00
if (!isValid()) return;
2022-03-01 00:03:28 +01:00
std::fwrite(bytes.data(), 1, bytes.size(), this->m_file);
}
void File::write(const std::string &string) {
2021-09-11 14:41:18 +02:00
if (!isValid()) return;
2022-03-01 00:03:28 +01:00
std::fwrite(string.data(), string.size(), 1, this->m_file);
}
2021-09-21 02:48:41 +02:00
size_t File::getSize() const {
2021-09-11 14:41:18 +02:00
if (!isValid()) return 0;
auto startPos = ftello64(this->m_file);
fseeko64(this->m_file, 0, SEEK_END);
auto size = ftello64(this->m_file);
fseeko64(this->m_file, startPos, SEEK_SET);
if (size < 0)
return 0;
return size;
}
void File::setSize(u64 size) {
2021-09-11 14:41:18 +02:00
if (!isValid()) return;
auto result = ftruncate64(fileno(this->m_file), size);
hex::unused(result);
}
void File::flush() {
2022-03-01 00:03:28 +01:00
std::fflush(this->m_file);
}
bool File::remove() {
this->close();
return std::remove(this->m_path.string().c_str()) == 0;
}
2022-03-01 00:03:28 +01:00
void File::disableBuffering() {
if (!isValid()) return;
std::setvbuf(this->m_file, nullptr, _IONBF, 0);
}
}