Device: Add disk write driver framework
At the moment, no drivers are implemented, so all devices have Access::None.v0.3.0-dev
parent
80cd4ae052
commit
0dcd950092
|
|
@ -149,7 +149,8 @@ set(SRC_FILES
|
||||||
device/devicefinder.cpp
|
device/devicefinder.cpp
|
||||||
device/device.cpp
|
device/device.cpp
|
||||||
disk/diskreaddriver.cpp
|
disk/diskreaddriver.cpp
|
||||||
disk/nulldiskreaddriver.cpp
|
disk/diskwritedriver.cpp
|
||||||
|
disk/nulldiskdriver.cpp
|
||||||
disk/neomemorydiskreaddriver.cpp
|
disk/neomemorydiskreaddriver.cpp
|
||||||
${PLATFORM_SRC}
|
${PLATFORM_SRC}
|
||||||
)
|
)
|
||||||
|
|
|
||||||
|
|
@ -107,6 +107,8 @@ static constexpr const char* ETH_PHY_REGISTER_CONTROL_NOT_AVAILABLE = "Ethernet
|
||||||
static constexpr const char* DISK_NOT_SUPPORTED = "This device does not support accessing the specified disk.";
|
static constexpr const char* DISK_NOT_SUPPORTED = "This device does not support accessing the specified disk.";
|
||||||
static constexpr const char* EOF_REACHED = "The requested length exceeds the available data from this disk.";
|
static constexpr const char* EOF_REACHED = "The requested length exceeds the available data from this disk.";
|
||||||
static constexpr const char* SETTINGS_DEFAULTS_USED = "The device settings could not be loaded, the default settings have been applied.";
|
static constexpr const char* SETTINGS_DEFAULTS_USED = "The device settings could not be loaded, the default settings have been applied.";
|
||||||
|
static constexpr const char* ATOMIC_OPERATION_RETRIED = "An operation failed to be atomically completed, but will be retried.";
|
||||||
|
static constexpr const char* ATOMIC_OPERATION_COMPLETED_NONATOMICALLY = "An ideally-atomic operation was completed nonatomically.";
|
||||||
|
|
||||||
// Transport Errors
|
// Transport Errors
|
||||||
static constexpr const char* FAILED_TO_READ = "A read operation failed.";
|
static constexpr const char* FAILED_TO_READ = "A read operation failed.";
|
||||||
|
|
@ -226,6 +228,10 @@ const char* APIEvent::DescriptionForType(Type type) {
|
||||||
return EOF_REACHED;
|
return EOF_REACHED;
|
||||||
case Type::SettingsDefaultsUsed:
|
case Type::SettingsDefaultsUsed:
|
||||||
return SETTINGS_DEFAULTS_USED;
|
return SETTINGS_DEFAULTS_USED;
|
||||||
|
case Type::AtomicOperationRetried:
|
||||||
|
return ATOMIC_OPERATION_RETRIED;
|
||||||
|
case Type::AtomicOperationCompletedNonatomically:
|
||||||
|
return ATOMIC_OPERATION_COMPLETED_NONATOMICALLY;
|
||||||
|
|
||||||
// Transport Errors
|
// Transport Errors
|
||||||
case Type::FailedToRead:
|
case Type::FailedToRead:
|
||||||
|
|
|
||||||
|
|
@ -488,6 +488,20 @@ optional<uint64_t> Device::readLogicalDisk(uint64_t pos, uint8_t* into, uint64_t
|
||||||
return diskReadDriver->readLogicalDisk(*com, report, pos, into, amount, timeout);
|
return diskReadDriver->readLogicalDisk(*com, report, pos, into, amount, timeout);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
optional<uint64_t> Device::writeLogicalDisk(uint64_t pos, const uint8_t* from, uint64_t amount, std::chrono::milliseconds timeout) {
|
||||||
|
if(!from || timeout <= std::chrono::milliseconds(0)) {
|
||||||
|
report(APIEvent::Type::RequiredParameterNull, APIEvent::Severity::Error);
|
||||||
|
return nullopt;
|
||||||
|
}
|
||||||
|
|
||||||
|
if(!isOpen()) {
|
||||||
|
report(APIEvent::Type::DeviceCurrentlyClosed, APIEvent::Severity::Error);
|
||||||
|
return nullopt;
|
||||||
|
}
|
||||||
|
|
||||||
|
return diskWriteDriver->writeLogicalDisk(*com, report, *diskReadDriver, pos, from, amount, timeout);
|
||||||
|
}
|
||||||
|
|
||||||
optional<bool> Device::getDigitalIO(IO type, size_t number /* = 1 */) {
|
optional<bool> Device::getDigitalIO(IO type, size_t number /* = 1 */) {
|
||||||
if(number == 0) { // Start counting from 1
|
if(number == 0) { // Start counting from 1
|
||||||
report(APIEvent::Type::ParameterOutOfRange, APIEvent::Severity::Error);
|
report(APIEvent::Type::ParameterOutOfRange, APIEvent::Severity::Error);
|
||||||
|
|
|
||||||
|
|
@ -2,8 +2,9 @@
|
||||||
#include <cstring>
|
#include <cstring>
|
||||||
|
|
||||||
using namespace icsneo;
|
using namespace icsneo;
|
||||||
|
using namespace icsneo::Disk;
|
||||||
|
|
||||||
optional<uint64_t> DiskReadDriver::readLogicalDisk(Communication& com, device_eventhandler_t report,
|
optional<uint64_t> ReadDriver::readLogicalDisk(Communication& com, device_eventhandler_t report,
|
||||||
uint64_t pos, uint8_t* into, uint64_t amount, std::chrono::milliseconds timeout) {
|
uint64_t pos, uint8_t* into, uint64_t amount, std::chrono::milliseconds timeout) {
|
||||||
optional<uint64_t> ret;
|
optional<uint64_t> ret;
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -0,0 +1,88 @@
|
||||||
|
#include "icsneo/disk/diskwritedriver.h"
|
||||||
|
#include <cstring>
|
||||||
|
|
||||||
|
using namespace icsneo;
|
||||||
|
using namespace icsneo::Disk;
|
||||||
|
|
||||||
|
const uint64_t WriteDriver::RetryAtomic = std::numeric_limits<uint64_t>::max();
|
||||||
|
const APIEvent::Severity WriteDriver::NonatomicSeverity = APIEvent::Severity::EventInfo;
|
||||||
|
|
||||||
|
optional<uint64_t> WriteDriver::writeLogicalDisk(Communication& com, device_eventhandler_t report, ReadDriver& readDriver,
|
||||||
|
uint64_t pos, const uint8_t* from, uint64_t amount, std::chrono::milliseconds timeout) {
|
||||||
|
optional<uint64_t> ret;
|
||||||
|
|
||||||
|
const uint32_t idealBlockSize = getBlockSizeBounds().second;
|
||||||
|
|
||||||
|
// Write from here if we need to read-modify-write a block
|
||||||
|
// That would be the case either if we don't want some at the
|
||||||
|
// beginning or end of the block.
|
||||||
|
std::vector<uint8_t> alignedWriteBuffer;
|
||||||
|
|
||||||
|
// Read to here, ideally this can be sent back to the device to
|
||||||
|
// ensure an operation is atomic
|
||||||
|
std::vector<uint8_t> atomicBuffer(idealBlockSize);
|
||||||
|
|
||||||
|
const uint64_t startBlock = pos / idealBlockSize;
|
||||||
|
const uint32_t posWithinFirstBlock = static_cast<uint32_t>(pos % idealBlockSize);
|
||||||
|
uint64_t blocks = amount / idealBlockSize + (amount % idealBlockSize ? 1 : 0);
|
||||||
|
if(blocks * idealBlockSize - posWithinFirstBlock < amount)
|
||||||
|
blocks++; // We need one more block to get the last partial block's worth
|
||||||
|
uint64_t blocksProcessed = 0;
|
||||||
|
|
||||||
|
while(blocksProcessed < blocks && timeout >= std::chrono::milliseconds::zero()) {
|
||||||
|
const uint64_t currentBlock = startBlock + blocksProcessed;
|
||||||
|
|
||||||
|
const uint64_t fromOffset = std::max<uint64_t>((blocksProcessed * idealBlockSize) - posWithinFirstBlock, 0);
|
||||||
|
const uint32_t posWithinCurrentBlock = (blocksProcessed ? 0 : posWithinFirstBlock);
|
||||||
|
uint32_t curAmt = idealBlockSize - posWithinCurrentBlock;
|
||||||
|
const auto amountLeft = amount - ret.value_or(0);
|
||||||
|
if(curAmt > amountLeft)
|
||||||
|
curAmt = static_cast<uint32_t>(amountLeft);
|
||||||
|
|
||||||
|
auto start = std::chrono::high_resolution_clock::now();
|
||||||
|
auto amount = readDriver.readLogicalDisk(com, report, currentBlock * idealBlockSize, atomicBuffer.data(),
|
||||||
|
idealBlockSize, timeout);
|
||||||
|
timeout -= std::chrono::duration_cast<std::chrono::milliseconds>(std::chrono::high_resolution_clock::now() - start);
|
||||||
|
|
||||||
|
if(amount != idealBlockSize) {
|
||||||
|
if(timeout < std::chrono::milliseconds::zero())
|
||||||
|
report(APIEvent::Type::Timeout, APIEvent::Severity::Error);
|
||||||
|
else
|
||||||
|
report(blocksProcessed ? APIEvent::Type::EOFReached : APIEvent::Type::ParameterOutOfRange, APIEvent::Severity::Error);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
const bool useAlignedWriteBuffer = (posWithinCurrentBlock != 0 || curAmt != idealBlockSize);
|
||||||
|
if(useAlignedWriteBuffer) {
|
||||||
|
if(alignedWriteBuffer.size() < idealBlockSize)
|
||||||
|
alignedWriteBuffer.resize(idealBlockSize);
|
||||||
|
memcpy(alignedWriteBuffer.data() + posWithinCurrentBlock, from + fromOffset, curAmt);
|
||||||
|
}
|
||||||
|
|
||||||
|
start = std::chrono::high_resolution_clock::now();
|
||||||
|
amount = writeLogicalDiskAligned(com, report, currentBlock * idealBlockSize, atomicBuffer.data(),
|
||||||
|
useAlignedWriteBuffer ? alignedWriteBuffer.data() : (from + fromOffset), idealBlockSize, timeout);
|
||||||
|
timeout -= std::chrono::duration_cast<std::chrono::milliseconds>(std::chrono::high_resolution_clock::now() - start);
|
||||||
|
|
||||||
|
if(amount == RetryAtomic) {
|
||||||
|
// The user may want to log these events in order to see how many atomic misses they are getting
|
||||||
|
report(APIEvent::Type::AtomicOperationRetried, APIEvent::Severity::EventInfo);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if(!amount.has_value() || *amount == 0) {
|
||||||
|
if(timeout < std::chrono::milliseconds::zero())
|
||||||
|
report(APIEvent::Type::Timeout, APIEvent::Severity::Error);
|
||||||
|
else
|
||||||
|
report(blocksProcessed ? APIEvent::Type::EOFReached : APIEvent::Type::ParameterOutOfRange, APIEvent::Severity::Error);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
if(!ret)
|
||||||
|
ret.emplace();
|
||||||
|
*ret += std::min<uint64_t>(*amount, curAmt);
|
||||||
|
blocksProcessed++;
|
||||||
|
}
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
@ -3,6 +3,7 @@
|
||||||
#include <cstring>
|
#include <cstring>
|
||||||
|
|
||||||
using namespace icsneo;
|
using namespace icsneo;
|
||||||
|
using namespace icsneo::Disk;
|
||||||
|
|
||||||
optional<uint64_t> NeoMemoryDiskReadDriver::readLogicalDiskAligned(Communication& com, device_eventhandler_t report,
|
optional<uint64_t> NeoMemoryDiskReadDriver::readLogicalDiskAligned(Communication& com, device_eventhandler_t report,
|
||||||
uint64_t pos, uint8_t* into, uint64_t amount, std::chrono::milliseconds timeout) {
|
uint64_t pos, uint8_t* into, uint64_t amount, std::chrono::milliseconds timeout) {
|
||||||
|
|
|
||||||
|
|
@ -0,0 +1,28 @@
|
||||||
|
#include "icsneo/disk/nulldiskdriver.h"
|
||||||
|
|
||||||
|
using namespace icsneo;
|
||||||
|
using namespace icsneo::Disk;
|
||||||
|
|
||||||
|
optional<uint64_t> NullDriver::readLogicalDisk(Communication&, device_eventhandler_t report,
|
||||||
|
uint64_t, uint8_t*, uint64_t, std::chrono::milliseconds) {
|
||||||
|
report(APIEvent::Type::DiskNotSupported, APIEvent::Severity::Error);
|
||||||
|
return nullopt;
|
||||||
|
}
|
||||||
|
|
||||||
|
optional<uint64_t> NullDriver::readLogicalDiskAligned(Communication&, device_eventhandler_t report,
|
||||||
|
uint64_t, uint8_t*, uint64_t, std::chrono::milliseconds) {
|
||||||
|
report(APIEvent::Type::DiskNotSupported, APIEvent::Severity::Error);
|
||||||
|
return nullopt;
|
||||||
|
}
|
||||||
|
|
||||||
|
optional<uint64_t> NullDriver::writeLogicalDisk(Communication&, device_eventhandler_t report, ReadDriver&,
|
||||||
|
uint64_t, const uint8_t*, uint64_t, std::chrono::milliseconds) {
|
||||||
|
report(APIEvent::Type::DiskNotSupported, APIEvent::Severity::Error);
|
||||||
|
return nullopt;
|
||||||
|
}
|
||||||
|
|
||||||
|
optional<uint64_t> NullDriver::writeLogicalDiskAligned(Communication&, device_eventhandler_t report,
|
||||||
|
uint64_t, const uint8_t*, const uint8_t*, uint64_t, std::chrono::milliseconds) {
|
||||||
|
report(APIEvent::Type::DiskNotSupported, APIEvent::Severity::Error);
|
||||||
|
return nullopt;
|
||||||
|
}
|
||||||
|
|
@ -1,15 +0,0 @@
|
||||||
#include "icsneo/disk/nulldiskreaddriver.h"
|
|
||||||
|
|
||||||
using namespace icsneo;
|
|
||||||
|
|
||||||
optional<uint64_t> NullDiskReadDriver::readLogicalDisk(Communication&, device_eventhandler_t report,
|
|
||||||
uint64_t, uint8_t*, uint64_t, std::chrono::milliseconds) {
|
|
||||||
report(APIEvent::Type::DiskNotSupported, APIEvent::Severity::Error);
|
|
||||||
return nullopt;
|
|
||||||
}
|
|
||||||
|
|
||||||
optional<uint64_t> NullDiskReadDriver::readLogicalDiskAligned(Communication&, device_eventhandler_t report,
|
|
||||||
uint64_t, uint8_t*, uint64_t, std::chrono::milliseconds) {
|
|
||||||
report(APIEvent::Type::DiskNotSupported, APIEvent::Severity::Error);
|
|
||||||
return nullopt;
|
|
||||||
}
|
|
||||||
|
|
@ -84,6 +84,8 @@ public:
|
||||||
DiskNotSupported = 0x2031,
|
DiskNotSupported = 0x2031,
|
||||||
EOFReached = 0x2032,
|
EOFReached = 0x2032,
|
||||||
SettingsDefaultsUsed = 0x2033,
|
SettingsDefaultsUsed = 0x2033,
|
||||||
|
AtomicOperationRetried = 0x2034,
|
||||||
|
AtomicOperationCompletedNonatomically = 0x2035,
|
||||||
|
|
||||||
// Transport Events
|
// Transport Events
|
||||||
FailedToRead = 0x3000,
|
FailedToRead = 0x3000,
|
||||||
|
|
|
||||||
|
|
@ -17,7 +17,8 @@
|
||||||
#include "icsneo/device/devicetype.h"
|
#include "icsneo/device/devicetype.h"
|
||||||
#include "icsneo/device/deviceversion.h"
|
#include "icsneo/device/deviceversion.h"
|
||||||
#include "icsneo/disk/diskreaddriver.h"
|
#include "icsneo/disk/diskreaddriver.h"
|
||||||
#include "icsneo/disk/nulldiskreaddriver.h"
|
#include "icsneo/disk/diskwritedriver.h"
|
||||||
|
#include "icsneo/disk/nulldiskdriver.h"
|
||||||
#include "icsneo/communication/communication.h"
|
#include "icsneo/communication/communication.h"
|
||||||
#include "icsneo/communication/packetizer.h"
|
#include "icsneo/communication/packetizer.h"
|
||||||
#include "icsneo/communication/encoder.h"
|
#include "icsneo/communication/encoder.h"
|
||||||
|
|
@ -156,7 +157,24 @@ public:
|
||||||
* set in icsneo::GetLastError().
|
* set in icsneo::GetLastError().
|
||||||
*/
|
*/
|
||||||
optional<uint64_t> readLogicalDisk(uint64_t pos, uint8_t* into, uint64_t amount,
|
optional<uint64_t> readLogicalDisk(uint64_t pos, uint8_t* into, uint64_t amount,
|
||||||
std::chrono::milliseconds timeout = DiskReadDriver::DefaultTimeout);
|
std::chrono::milliseconds timeout = Disk::DefaultTimeout);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Write to the logical disk in this device, starting from byte `pos`
|
||||||
|
* and writing up to `amount` bytes.
|
||||||
|
*
|
||||||
|
* The number of bytes written will be returned in case of success.
|
||||||
|
*
|
||||||
|
* If the number of bytes written is less than the amount requested,
|
||||||
|
* an error will be set in icsneo::GetLastError() explaining why.
|
||||||
|
* Likely, either the end of the logical disk has been reached, or
|
||||||
|
* the timeout was reached while the write had only partially completed.
|
||||||
|
*
|
||||||
|
* Upon failure, icsneo::nullopt will be returned and an error will be
|
||||||
|
* set in icsneo::GetLastError().
|
||||||
|
*/
|
||||||
|
optional<uint64_t> writeLogicalDisk(uint64_t pos, const uint8_t* from, uint64_t amount,
|
||||||
|
std::chrono::milliseconds timeout = Disk::DefaultTimeout);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Retrieve the number of Ethernet (DoIP) Activation lines present
|
* Retrieve the number of Ethernet (DoIP) Activation lines present
|
||||||
|
|
@ -276,7 +294,7 @@ protected:
|
||||||
data.device = this;
|
data.device = this;
|
||||||
}
|
}
|
||||||
|
|
||||||
template<typename Driver, typename Settings = NullSettings, typename DiskRead = NullDiskReadDriver>
|
template<typename Driver, typename Settings = NullSettings, typename DiskRead = Disk::NullDriver, typename DiskWrite = Disk::NullDriver>
|
||||||
void initialize() {
|
void initialize() {
|
||||||
report = makeEventHandler();
|
report = makeEventHandler();
|
||||||
auto driver = makeDriver<Driver>();
|
auto driver = makeDriver<Driver>();
|
||||||
|
|
@ -289,7 +307,8 @@ protected:
|
||||||
setupCommunication(*com);
|
setupCommunication(*com);
|
||||||
settings = makeSettings<Settings>(com);
|
settings = makeSettings<Settings>(com);
|
||||||
setupSettings(*settings);
|
setupSettings(*settings);
|
||||||
diskReadDriver = std::make_unique<DiskRead>();
|
diskReadDriver = std::unique_ptr<DiskRead>(new DiskRead());
|
||||||
|
diskWriteDriver = std::unique_ptr<DiskWrite>(new DiskWrite());
|
||||||
setupSupportedRXNetworks(supportedRXNetworks);
|
setupSupportedRXNetworks(supportedRXNetworks);
|
||||||
setupSupportedTXNetworks(supportedTXNetworks);
|
setupSupportedTXNetworks(supportedTXNetworks);
|
||||||
setupExtensions();
|
setupExtensions();
|
||||||
|
|
@ -365,7 +384,8 @@ private:
|
||||||
neodevice_t data;
|
neodevice_t data;
|
||||||
std::shared_ptr<ResetStatusMessage> latestResetStatus;
|
std::shared_ptr<ResetStatusMessage> latestResetStatus;
|
||||||
std::vector<optional<DeviceAppVersion>> versions;
|
std::vector<optional<DeviceAppVersion>> versions;
|
||||||
std::unique_ptr<DiskReadDriver> diskReadDriver;
|
std::unique_ptr<Disk::ReadDriver> diskReadDriver;
|
||||||
|
std::unique_ptr<Disk::WriteDriver> diskWriteDriver;
|
||||||
|
|
||||||
mutable std::mutex extensionsLock;
|
mutable std::mutex extensionsLock;
|
||||||
std::vector<std::shared_ptr<DeviceExtension>> extensions;
|
std::vector<std::shared_ptr<DeviceExtension>> extensions;
|
||||||
|
|
|
||||||
|
|
@ -0,0 +1,38 @@
|
||||||
|
#ifndef __DISKDRIVER_H__
|
||||||
|
#define __DISKDRIVER_H__
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
|
||||||
|
#include <cstdint>
|
||||||
|
#include <chrono>
|
||||||
|
#include <utility>
|
||||||
|
|
||||||
|
namespace icsneo {
|
||||||
|
|
||||||
|
namespace Disk {
|
||||||
|
|
||||||
|
constexpr const std::chrono::milliseconds DefaultTimeout{2000};
|
||||||
|
constexpr const size_t SectorSize = 512;
|
||||||
|
enum class Access {
|
||||||
|
None,
|
||||||
|
EntireCard,
|
||||||
|
VSA
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Interface for drivers which work with block data on devices
|
||||||
|
*/
|
||||||
|
class Driver {
|
||||||
|
public:
|
||||||
|
virtual ~Driver() = default;
|
||||||
|
virtual Access getAccess() const = 0;
|
||||||
|
virtual std::pair<uint32_t, uint32_t> getBlockSizeBounds() const = 0;
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace Disk
|
||||||
|
|
||||||
|
} // namespace icsneo
|
||||||
|
|
||||||
|
#endif // __cplusplus
|
||||||
|
|
||||||
|
#endif // __DISKDRIVER_H__
|
||||||
|
|
@ -6,29 +6,21 @@
|
||||||
#include "icsneo/platform/optional.h"
|
#include "icsneo/platform/optional.h"
|
||||||
#include "icsneo/communication/communication.h"
|
#include "icsneo/communication/communication.h"
|
||||||
#include "icsneo/api/eventmanager.h"
|
#include "icsneo/api/eventmanager.h"
|
||||||
|
#include "icsneo/disk/diskdriver.h"
|
||||||
#include <cstdint>
|
#include <cstdint>
|
||||||
#include <chrono>
|
#include <chrono>
|
||||||
|
|
||||||
namespace icsneo {
|
namespace icsneo {
|
||||||
|
|
||||||
|
namespace Disk {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Interface for drivers which read block data from devices
|
* Interface for drivers which read block data from devices
|
||||||
*/
|
*/
|
||||||
class DiskReadDriver {
|
class ReadDriver : public virtual Driver {
|
||||||
public:
|
public:
|
||||||
static constexpr const std::chrono::milliseconds DefaultTimeout{2000};
|
|
||||||
static constexpr const size_t SectorSize = 512;
|
|
||||||
enum class Access {
|
|
||||||
None,
|
|
||||||
EntireCard,
|
|
||||||
VSA
|
|
||||||
};
|
|
||||||
|
|
||||||
virtual ~DiskReadDriver() = default;
|
|
||||||
virtual optional<uint64_t> readLogicalDisk(Communication& com, device_eventhandler_t report,
|
virtual optional<uint64_t> readLogicalDisk(Communication& com, device_eventhandler_t report,
|
||||||
uint64_t pos, uint8_t* into, uint64_t amount, std::chrono::milliseconds timeout = DefaultTimeout);
|
uint64_t pos, uint8_t* into, uint64_t amount, std::chrono::milliseconds timeout = DefaultTimeout);
|
||||||
virtual Access getAccess() const = 0;
|
|
||||||
virtual std::pair<uint32_t, uint32_t> getBlockSizeBounds() const = 0;
|
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
/**
|
/**
|
||||||
|
|
@ -41,7 +33,9 @@ protected:
|
||||||
uint64_t pos, uint8_t* into, uint64_t amount, std::chrono::milliseconds timeout) = 0;
|
uint64_t pos, uint8_t* into, uint64_t amount, std::chrono::milliseconds timeout) = 0;
|
||||||
};
|
};
|
||||||
|
|
||||||
}
|
} // namespace Disk
|
||||||
|
|
||||||
|
} // namespace icsneo
|
||||||
|
|
||||||
#endif // __cplusplus
|
#endif // __cplusplus
|
||||||
#endif // __DISKREADDRIVER_H__
|
#endif // __DISKREADDRIVER_H__
|
||||||
|
|
@ -0,0 +1,65 @@
|
||||||
|
#ifndef __DISKWRITEDRIVER_H__
|
||||||
|
#define __DISKWRITEDRIVER_H__
|
||||||
|
|
||||||
|
#ifdef __cplusplus
|
||||||
|
|
||||||
|
#include "icsneo/platform/optional.h"
|
||||||
|
#include "icsneo/communication/communication.h"
|
||||||
|
#include "icsneo/api/eventmanager.h"
|
||||||
|
#include "icsneo/disk/diskreaddriver.h"
|
||||||
|
#include <cstdint>
|
||||||
|
#include <chrono>
|
||||||
|
|
||||||
|
namespace icsneo {
|
||||||
|
|
||||||
|
namespace Disk {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Interface for drivers which write block data from devices
|
||||||
|
*/
|
||||||
|
class WriteDriver : public virtual Driver {
|
||||||
|
public:
|
||||||
|
virtual optional<uint64_t> writeLogicalDisk(Communication& com, device_eventhandler_t report, ReadDriver& readDriver,
|
||||||
|
uint64_t pos, const uint8_t* from, uint64_t amount, std::chrono::milliseconds timeout = DefaultTimeout);
|
||||||
|
|
||||||
|
protected:
|
||||||
|
/**
|
||||||
|
* Flag returned from writeLogicalDiskAligned when the
|
||||||
|
* operation failed to be performed atomically and can
|
||||||
|
* be retried after rereading.
|
||||||
|
*/
|
||||||
|
static const uint64_t RetryAtomic;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The severity to report with when an atomic operation
|
||||||
|
* is requested that the driver is unable to attempt.
|
||||||
|
*/
|
||||||
|
static const APIEvent::Severity NonatomicSeverity;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Perform a write which the driver can do in one shot.
|
||||||
|
*
|
||||||
|
* The `pos` requested must be sector-aligned, and the `amount` must be
|
||||||
|
* within the block size bounds provided by the driver.
|
||||||
|
*
|
||||||
|
* If `atomicBuf` is provided, it will be used to ensure that the disk
|
||||||
|
* data changes from `atomicBuf` to `from` without trampling any reads
|
||||||
|
* that may have happened while modifying the data.
|
||||||
|
*
|
||||||
|
* The flag `RetryAtomic` is returned if the operation was attempted
|
||||||
|
* atomically but failed.
|
||||||
|
*
|
||||||
|
* If the driver does not support atomic operations, but `atomicBuf`
|
||||||
|
* is non-null, an APIEvent::AtomicOperationCompletedNonatomically
|
||||||
|
* should be reported with `NonatomicSeverity`.
|
||||||
|
*/
|
||||||
|
virtual optional<uint64_t> writeLogicalDiskAligned(Communication& com, device_eventhandler_t report,
|
||||||
|
uint64_t pos, const uint8_t* atomicBuf, const uint8_t* from, uint64_t amount, std::chrono::milliseconds timeout) = 0;
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace Disk
|
||||||
|
|
||||||
|
} // namespace icsneo
|
||||||
|
|
||||||
|
#endif // __cplusplus
|
||||||
|
#endif // __DISKWRITEDRIVER_H__
|
||||||
|
|
@ -8,12 +8,14 @@
|
||||||
|
|
||||||
namespace icsneo {
|
namespace icsneo {
|
||||||
|
|
||||||
|
namespace Disk {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A disk read driver which uses the neoMemory command to read from the disk
|
* A disk read driver which uses the neoMemory command to read from the disk
|
||||||
*
|
*
|
||||||
* This can only request reads by sector, so it will be very slow, but is likely supported by any device with a disk
|
* This can only request reads by sector, so it will be very slow, but is likely supported by any device with a disk
|
||||||
*/
|
*/
|
||||||
class NeoMemoryDiskReadDriver : public DiskReadDriver {
|
class NeoMemoryDiskReadDriver : public ReadDriver {
|
||||||
public:
|
public:
|
||||||
Access getAccess() const override { return Access::VSA; }
|
Access getAccess() const override { return Access::VSA; }
|
||||||
std::pair<uint32_t, uint32_t> getBlockSizeBounds() const override {
|
std::pair<uint32_t, uint32_t> getBlockSizeBounds() const override {
|
||||||
|
|
@ -29,7 +31,9 @@ private:
|
||||||
uint64_t pos, uint8_t* into, uint64_t amount, std::chrono::milliseconds timeout) override;
|
uint64_t pos, uint8_t* into, uint64_t amount, std::chrono::milliseconds timeout) override;
|
||||||
};
|
};
|
||||||
|
|
||||||
}
|
} // namespace Disk
|
||||||
|
|
||||||
|
} // namespace icsneo
|
||||||
|
|
||||||
#endif // __cplusplus
|
#endif // __cplusplus
|
||||||
#endif // __NEOMEMORYDISKREADDRIVER_H__
|
#endif // __NEOMEMORYDISKREADDRIVER_H__
|
||||||
|
|
@ -4,19 +4,24 @@
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
|
|
||||||
#include "icsneo/disk/diskreaddriver.h"
|
#include "icsneo/disk/diskreaddriver.h"
|
||||||
|
#include "icsneo/disk/diskwritedriver.h"
|
||||||
#include <limits>
|
#include <limits>
|
||||||
|
|
||||||
namespace icsneo {
|
namespace icsneo {
|
||||||
|
|
||||||
|
namespace Disk {
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A disk driver which always returns the requested disk as unsupported
|
* A disk driver which always returns the requested disk as unsupported
|
||||||
*
|
*
|
||||||
* Used for devices which do not have a disk, or do not provide any means for accessing it
|
* Used for devices which do not have a disk, or do not provide any means for accessing it
|
||||||
*/
|
*/
|
||||||
class NullDiskReadDriver : public DiskReadDriver {
|
class NullDriver : public ReadDriver, public WriteDriver {
|
||||||
public:
|
public:
|
||||||
optional<uint64_t> readLogicalDisk(Communication& com, device_eventhandler_t report,
|
optional<uint64_t> readLogicalDisk(Communication& com, device_eventhandler_t report,
|
||||||
uint64_t pos, uint8_t* into, uint64_t amount, std::chrono::milliseconds timeout = DefaultTimeout) override;
|
uint64_t pos, uint8_t* into, uint64_t amount, std::chrono::milliseconds timeout = DefaultTimeout) override;
|
||||||
|
optional<uint64_t> writeLogicalDisk(Communication& com, device_eventhandler_t report, ReadDriver& readDriver,
|
||||||
|
uint64_t pos, const uint8_t* from, uint64_t amount, std::chrono::milliseconds timeout = DefaultTimeout) override;
|
||||||
Access getAccess() const override { return Access::None; }
|
Access getAccess() const override { return Access::None; }
|
||||||
std::pair<uint32_t, uint32_t> getBlockSizeBounds() const override {
|
std::pair<uint32_t, uint32_t> getBlockSizeBounds() const override {
|
||||||
static_assert(SectorSize <= std::numeric_limits<uint32_t>::max(), "Incorrect sector size");
|
static_assert(SectorSize <= std::numeric_limits<uint32_t>::max(), "Incorrect sector size");
|
||||||
|
|
@ -27,9 +32,13 @@ public:
|
||||||
private:
|
private:
|
||||||
optional<uint64_t> readLogicalDiskAligned(Communication& com, device_eventhandler_t report,
|
optional<uint64_t> readLogicalDiskAligned(Communication& com, device_eventhandler_t report,
|
||||||
uint64_t pos, uint8_t* into, uint64_t amount, std::chrono::milliseconds timeout) override;
|
uint64_t pos, uint8_t* into, uint64_t amount, std::chrono::milliseconds timeout) override;
|
||||||
|
optional<uint64_t> writeLogicalDiskAligned(Communication& com, device_eventhandler_t report,
|
||||||
|
uint64_t pos, const uint8_t* atomicBuf, const uint8_t* from, uint64_t amount, std::chrono::milliseconds timeout) override;
|
||||||
};
|
};
|
||||||
|
|
||||||
}
|
} // namespace Disk
|
||||||
|
|
||||||
|
} // namespace icsneo
|
||||||
|
|
||||||
#endif // __cplusplus
|
#endif // __cplusplus
|
||||||
#endif // __NULLDISKREADDRIVER_H__
|
#endif // __NULLDISKREADDRIVER_H__
|
||||||
Loading…
Reference in New Issue