2022-10-23 19:51:39 +00:00
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
//
|
|
|
|
// SCSI Target Emulator RaSCSI Reloaded
|
|
|
|
// for Raspberry Pi
|
|
|
|
//
|
|
|
|
// Copyright (C) 2022 Uwe Seimet
|
|
|
|
//
|
|
|
|
//---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
#include "rascsi_exceptions.h"
|
|
|
|
#include "storage_device.h"
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <filesystem>
|
|
|
|
|
|
|
|
using namespace std;
|
|
|
|
using namespace filesystem;
|
|
|
|
|
|
|
|
unordered_map<string, id_set> StorageDevice::reserved_files;
|
|
|
|
|
|
|
|
StorageDevice::StorageDevice(PbDeviceType type, int lun) : ModePageDevice(type, lun)
|
|
|
|
{
|
|
|
|
SupportsFile(true);
|
|
|
|
SetStoppable(true);
|
|
|
|
}
|
|
|
|
|
2022-10-25 08:29:57 +00:00
|
|
|
void StorageDevice::ValidateFile()
|
2022-10-23 19:51:39 +00:00
|
|
|
{
|
|
|
|
if (blocks == 0) {
|
|
|
|
throw io_exception(string(GetTypeString()) + " device has 0 blocks");
|
|
|
|
}
|
|
|
|
|
2022-10-25 08:29:57 +00:00
|
|
|
if (!exists(path(filename))) {
|
|
|
|
throw file_not_found_exception("Image file '" + filename + "' for " + GetTypeString() + " device does not exist");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (GetFileSize() > 2LL * 1024 * 1024 * 1024 * 1024) {
|
|
|
|
throw io_exception("Drive capacity cannot exceed 2 TiB");
|
|
|
|
}
|
|
|
|
|
2022-10-23 19:51:39 +00:00
|
|
|
// TODO Check for duplicate handling of these properties (-> rascsi_executor.cpp)
|
2022-10-25 08:29:57 +00:00
|
|
|
if (access(filename.c_str(), W_OK)) {
|
2022-10-23 19:51:39 +00:00
|
|
|
// Permanently write-protected
|
|
|
|
SetReadOnly(true);
|
|
|
|
SetProtectable(false);
|
|
|
|
SetProtected(false);
|
|
|
|
}
|
|
|
|
|
|
|
|
SetStopped(false);
|
|
|
|
SetRemoved(false);
|
|
|
|
SetLocked(false);
|
|
|
|
SetReady(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
void StorageDevice::MediumChanged()
|
|
|
|
{
|
|
|
|
assert(IsRemovable());
|
|
|
|
|
|
|
|
medium_changed = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void StorageDevice::ReserveFile(const string& file, int id, int lun) const
|
|
|
|
{
|
|
|
|
assert(!file.empty());
|
|
|
|
assert(reserved_files.find(file) == reserved_files.end());
|
|
|
|
|
|
|
|
reserved_files[file] = make_pair(id, lun);
|
|
|
|
}
|
|
|
|
|
|
|
|
void StorageDevice::UnreserveFile()
|
|
|
|
{
|
|
|
|
reserved_files.erase(filename);
|
|
|
|
|
|
|
|
filename = "";
|
|
|
|
}
|
|
|
|
|
2022-10-25 08:29:57 +00:00
|
|
|
pair<int, int> StorageDevice::GetIdsForReservedFile(const string& file)
|
2022-10-23 19:51:39 +00:00
|
|
|
{
|
|
|
|
if (const auto& it = reserved_files.find(file); it != reserved_files.end()) {
|
2022-10-25 08:29:57 +00:00
|
|
|
return make_pair(it->second.first, it->second.second);
|
2022-10-23 19:51:39 +00:00
|
|
|
}
|
|
|
|
|
2022-10-25 08:29:57 +00:00
|
|
|
return make_pair(-1, -1);
|
2022-10-23 19:51:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void StorageDevice::UnreserveAll()
|
|
|
|
{
|
|
|
|
reserved_files.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool StorageDevice::FileExists(const string& file)
|
|
|
|
{
|
|
|
|
return exists(file);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool StorageDevice::IsReadOnlyFile() const
|
|
|
|
{
|
|
|
|
return access(filename.c_str(), W_OK);
|
|
|
|
}
|
|
|
|
|
|
|
|
off_t StorageDevice::GetFileSize() const
|
|
|
|
{
|
2022-10-25 08:29:57 +00:00
|
|
|
// filesystem::file_size cannot be used here because gcc < 10.3.0 cannot handle more than 2 GiB
|
2022-10-23 19:51:39 +00:00
|
|
|
if (struct stat st; !stat(filename.c_str(), &st)) {
|
|
|
|
return st.st_size;
|
|
|
|
}
|
|
|
|
|
|
|
|
throw io_exception("Can't get size of '" + filename + "'");
|
|
|
|
}
|