Device: Add disk write driver framework

At the moment, no drivers are implemented, so all devices have Access::None.
v0.3.0-dev
Paul Hollinsky 2022-02-24 19:35:06 -05:00
parent 80cd4ae052
commit 0dcd950092
15 changed files with 295 additions and 39 deletions

View File

@ -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}
) )

View File

@ -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:

View File

@ -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);

View File

@ -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;

View File

@ -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;
}

View File

@ -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) {

View File

@ -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;
}

View File

@ -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;
}

View File

@ -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,

View File

@ -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;

View File

@ -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__

View File

@ -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__

View File

@ -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__

View File

@ -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__

View File

@ -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__