FS helpers #604
14
CHANGELOG.md
14
CHANGELOG.md
@ -16,6 +16,20 @@ will consitute of a breaking change warranting a new major release:
|
|||||||
|
|
||||||
# [unreleased]
|
# [unreleased]
|
||||||
|
|
||||||
|
## Added
|
||||||
|
|
||||||
|
- Added `mv`, `cp` and `rm` action helpers for the core controller for common filesystem operations.
|
||||||
|
- Extended directory listing helpers. There is now a directory listing helper which dumps the
|
||||||
|
directory listing as an action data reply immediately. For smaller directory listings, this
|
||||||
|
allows a listing without requiring a separate file downlink (which also has not been implemented
|
||||||
|
yet)
|
||||||
|
|
||||||
|
## Changed
|
||||||
|
|
||||||
|
- The directory listing action commands now allow compressing of either the target output file
|
||||||
|
for the directory listing into file action command, or compression in the helper which dumps
|
||||||
|
the directory listing directly.
|
||||||
|
|
||||||
# [v1.45.0] 2023-04-14
|
# [v1.45.0] 2023-04-14
|
||||||
|
|
||||||
- q7s-package: v2.5.0
|
- q7s-package: v2.5.0
|
||||||
|
@ -233,6 +233,67 @@ ReturnValue_t CoreController::executeAction(ActionId_t actionId, MessageQueueId_
|
|||||||
case (LIST_DIRECTORY_INTO_FILE): {
|
case (LIST_DIRECTORY_INTO_FILE): {
|
||||||
return actionListDirectoryIntoFile(actionId, commandedBy, data, size);
|
return actionListDirectoryIntoFile(actionId, commandedBy, data, size);
|
||||||
}
|
}
|
||||||
|
case (LIST_DIRECTORY_DUMP_DIRECTLY): {
|
||||||
|
return actionListDirectoryDumpDirectly(actionId, commandedBy, data, size);
|
||||||
|
}
|
||||||
|
case (CP_HELPER): {
|
||||||
|
CpHelperParser parser(data, size);
|
||||||
|
ReturnValue_t result = parser.parse();
|
||||||
|
if (result != returnvalue::OK) {
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
std::ostringstream oss("cp ", std::ostringstream::ate);
|
||||||
|
if (parser.isRecursiveOptSet()) {
|
||||||
|
oss << "-r ";
|
||||||
|
}
|
||||||
|
auto &sourceTgt = parser.destTgtPair();
|
||||||
|
oss << sourceTgt.sourceName << " " << sourceTgt.targetName;
|
||||||
|
int ret = std::system(oss.str().c_str());
|
||||||
|
if (ret != 0) {
|
||||||
|
return returnvalue::FAILED;
|
||||||
|
}
|
||||||
|
return EXECUTION_FINISHED;
|
||||||
|
}
|
||||||
|
case (MV_HELPER): {
|
||||||
|
MvHelperParser parser(data, size);
|
||||||
|
ReturnValue_t result = parser.parse();
|
||||||
|
if (result != returnvalue::OK) {
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
std::ostringstream oss("mv ", std::ostringstream::ate);
|
||||||
|
auto &sourceTgt = parser.destTgtPair();
|
||||||
|
oss << sourceTgt.sourceName << " " << sourceTgt.targetName;
|
||||||
|
int ret = std::system(oss.str().c_str());
|
||||||
|
if (ret != 0) {
|
||||||
|
return returnvalue::FAILED;
|
||||||
|
}
|
||||||
|
return EXECUTION_FINISHED;
|
||||||
|
}
|
||||||
|
case (RM_HELPER): {
|
||||||
|
RmHelperParser parser(data, size);
|
||||||
|
ReturnValue_t result = parser.parse();
|
||||||
|
if (result != returnvalue::OK) {
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
std::ostringstream oss("rm ", std::ostringstream::ate);
|
||||||
|
if (parser.isRecursiveOptSet() or parser.isForceOptSet()) {
|
||||||
|
oss << "-";
|
||||||
|
}
|
||||||
|
if (parser.isRecursiveOptSet()) {
|
||||||
|
oss << "r";
|
||||||
|
}
|
||||||
|
if (parser.isForceOptSet()) {
|
||||||
|
oss << "f";
|
||||||
|
}
|
||||||
|
size_t removeTargetSize = 0;
|
||||||
|
const char *removeTgt = parser.getRemoveTarget(removeTargetSize);
|
||||||
|
oss << " " << removeTgt;
|
||||||
|
int ret = std::system(oss.str().c_str());
|
||||||
|
if (ret != 0) {
|
||||||
|
return returnvalue::FAILED;
|
||||||
|
}
|
||||||
|
return EXECUTION_FINISHED;
|
||||||
|
}
|
||||||
case (SWITCH_REBOOT_FILE_HANDLING): {
|
case (SWITCH_REBOOT_FILE_HANDLING): {
|
||||||
if (size < 1) {
|
if (size < 1) {
|
||||||
return HasActionsIF::INVALID_PARAMETERS;
|
return HasActionsIF::INVALID_PARAMETERS;
|
||||||
@ -911,57 +972,115 @@ ReturnValue_t CoreController::initVersionFile() {
|
|||||||
return returnvalue::OK;
|
return returnvalue::OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
ReturnValue_t CoreController::actionListDirectoryIntoFile(ActionId_t actionId,
|
ReturnValue_t CoreController::actionListDirectoryDumpDirectly(ActionId_t actionId,
|
||||||
MessageQueueId_t commandedBy,
|
MessageQueueId_t commandedBy,
|
||||||
const uint8_t *data, size_t size) {
|
const uint8_t *data, size_t size) {
|
||||||
// TODO: Packet definition for clean deserialization
|
core::ListDirectoryCmdBase parser(data, size);
|
||||||
// 2 bytes for a and R flag, at least 5 bytes for minimum valid path /tmp with
|
ReturnValue_t result = parser.parse();
|
||||||
// null termination, at least 7 bytes for minimum target file name /tmp/a with
|
if (result != returnvalue::OK) {
|
||||||
// null termination.
|
return result;
|
||||||
if (size < 14) {
|
|
||||||
return HasActionsIF::INVALID_PARAMETERS;
|
|
||||||
}
|
}
|
||||||
// We could also make -l optional, but I can't think of a reason why to not use -l..
|
|
||||||
|
|
||||||
// This flag specifies to run ls with -a
|
std::ostringstream oss("ls -l", std::ostringstream::ate);
|
||||||
bool aFlag = data[0];
|
|
||||||
data += 1;
|
|
||||||
// This flag specifies to run ls with -R
|
|
||||||
bool RFlag = data[1];
|
|
||||||
data += 1;
|
|
||||||
|
|
||||||
size_t remainingSize = size - 2;
|
|
||||||
// One larger for null termination, which prevents undefined behaviour if the sent
|
|
||||||
// strings are not 0 terminated properly
|
|
||||||
std::vector<uint8_t> repoAndTargetFileBuffer(remainingSize + 1, 0);
|
|
||||||
std::memcpy(repoAndTargetFileBuffer.data(), data, remainingSize);
|
|
||||||
const char *currentCharPtr = reinterpret_cast<const char *>(repoAndTargetFileBuffer.data());
|
|
||||||
// Full target file name
|
|
||||||
std::string repoName(currentCharPtr);
|
|
||||||
size_t repoLength = repoName.length();
|
|
||||||
// The other string needs to be at least one letter plus NULL termination to be valid at all
|
|
||||||
// The first string also needs to be NULL terminated, but the termination is not included
|
|
||||||
// in the string length, so this is subtracted from the remaining size as well
|
|
||||||
if (repoLength > remainingSize - 3) {
|
|
||||||
return HasActionsIF::INVALID_PARAMETERS;
|
|
||||||
}
|
|
||||||
// The file length will not include the NULL termination, so we skip it
|
|
||||||
currentCharPtr += repoLength + 1;
|
|
||||||
std::string targetFileName(currentCharPtr);
|
|
||||||
std::ostringstream oss;
|
|
||||||
oss << "ls -l";
|
oss << "ls -l";
|
||||||
if (aFlag) {
|
if (parser.aFlagSet()) {
|
||||||
oss << "a";
|
oss << "a";
|
||||||
}
|
}
|
||||||
if (RFlag) {
|
if (parser.rFlagSet()) {
|
||||||
oss << "R";
|
oss << "R";
|
||||||
}
|
}
|
||||||
|
|
||||||
|
size_t repoNameLen = 0;
|
||||||
|
const char *repoName = parser.getRepoName(repoNameLen);
|
||||||
|
|
||||||
|
oss << " " << repoName << " > " << LIST_DIR_DUMP_WORK_FILE;
|
||||||
|
int ret = std::system(oss.str().c_str());
|
||||||
|
if (ret != 0) {
|
||||||
|
utility::handleSystemError(result, "CoreController::actionListDirectoryDumpDirectly");
|
||||||
|
return returnvalue::FAILED;
|
||||||
|
}
|
||||||
|
if (parser.compressionOptionSet()) {
|
||||||
|
std::string compressedName = LIST_DIR_DUMP_WORK_FILE + std::string(".tar.xz");
|
||||||
|
oss.str("");
|
||||||
|
oss << "tar -cJvf /tmp/dir_listing_compressed.tmp " << LIST_DIR_DUMP_WORK_FILE;
|
||||||
|
ret = std::system(oss.str().c_str());
|
||||||
|
if (ret != 0) {
|
||||||
|
utility::handleSystemError(result, "CoreController::actionListDirectoryDumpDirectly");
|
||||||
|
return returnvalue::FAILED;
|
||||||
|
}
|
||||||
|
oss.str("");
|
||||||
|
// Overwrite the old work file with the compressed archive.
|
||||||
|
oss << "mv /tmp/dir_listing_compressed.tmp " << LIST_DIR_DUMP_WORK_FILE;
|
||||||
|
ret = std::system(oss.str().c_str());
|
||||||
|
if (ret != 0) {
|
||||||
|
utility::handleSystemError(result, "CoreController::actionListDirectoryDumpDirectly");
|
||||||
|
return returnvalue::FAILED;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
std::array<uint8_t, 1024> dirListingBuf{};
|
||||||
|
std::ifstream ifile("/tmp/dir_listing.tmp", std::ios::binary);
|
||||||
|
std::error_code e;
|
||||||
|
size_t totalFileSize = std::filesystem::file_size(LIST_DIR_DUMP_WORK_FILE, e);
|
||||||
|
size_t dumpedBytes = 0;
|
||||||
|
size_t nextDumpLen = 0;
|
||||||
|
while (dumpedBytes < totalFileSize) {
|
||||||
|
nextDumpLen = 1024;
|
||||||
|
if (totalFileSize - dumpedBytes < 1024) {
|
||||||
|
nextDumpLen = totalFileSize - dumpedBytes;
|
||||||
|
}
|
||||||
|
ifile.read(reinterpret_cast<char *>(dirListingBuf.data()), nextDumpLen);
|
||||||
|
result = actionHelper.reportData(commandedBy, actionId, dirListingBuf.data(), nextDumpLen);
|
||||||
|
if (result != returnvalue::OK) {
|
||||||
|
// Remove work file when we are done
|
||||||
|
std::filesystem::remove(LIST_DIR_DUMP_WORK_FILE, e);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Remove work file when we are done
|
||||||
|
std::filesystem::remove(LIST_DIR_DUMP_WORK_FILE, e);
|
||||||
|
return returnvalue::OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
ReturnValue_t CoreController::actionListDirectoryIntoFile(ActionId_t actionId,
|
||||||
|
MessageQueueId_t commandedBy,
|
||||||
|
const uint8_t *data, size_t size) {
|
||||||
|
core::ListDirectoryIntoFile parser(data, size);
|
||||||
|
ReturnValue_t result = parser.parse();
|
||||||
|
if (result != returnvalue::OK) {
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::ostringstream oss("ls -l", std::ostringstream::ate);
|
||||||
|
oss << "ls -l";
|
||||||
|
if (parser.aFlagSet()) {
|
||||||
|
oss << "a";
|
||||||
|
}
|
||||||
|
if (parser.rFlagSet()) {
|
||||||
|
oss << "R";
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t repoNameLen = 0;
|
||||||
|
const char *repoName = parser.getRepoName(repoNameLen);
|
||||||
|
size_t targetFileNameLen = 0;
|
||||||
|
const char *targetFileName = parser.getTargetName(targetFileNameLen);
|
||||||
oss << " " << repoName << " > " << targetFileName;
|
oss << " " << repoName << " > " << targetFileName;
|
||||||
int result = std::system(oss.str().c_str());
|
sif::info << "Executing list directory request, command: " << oss.str() << std::endl;
|
||||||
if (result != 0) {
|
int ret = std::system(oss.str().c_str());
|
||||||
|
if (ret != 0) {
|
||||||
utility::handleSystemError(result, "CoreController::actionListDirectoryIntoFile");
|
utility::handleSystemError(result, "CoreController::actionListDirectoryIntoFile");
|
||||||
actionHelper.finish(false, commandedBy, actionId);
|
return returnvalue::FAILED;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (parser.compressionOptionSet()) {
|
||||||
|
std::string compressedName = targetFileName + std::string(".tar.xz");
|
||||||
|
oss.str("");
|
||||||
|
oss << "tar -cJvf " << compressedName << " " << targetFileName;
|
||||||
|
sif::info << "Compressing directory listing: " << oss.str() << std::endl;
|
||||||
|
ret = std::system(oss.str().c_str());
|
||||||
|
if (ret != 0) {
|
||||||
|
utility::handleSystemError(result, "CoreController::actionListDirectoryIntoFile");
|
||||||
|
return returnvalue::FAILED;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return returnvalue::OK;
|
return returnvalue::OK;
|
||||||
}
|
}
|
||||||
|
@ -66,6 +66,7 @@ class CoreController : public ExtendedControllerBase, public ReceivesParameterMe
|
|||||||
static constexpr char CHIP_0_COPY_1_MOUNT_DIR[] = "/tmp/mntupdate-xdi-qspi0-gold-rootfs";
|
static constexpr char CHIP_0_COPY_1_MOUNT_DIR[] = "/tmp/mntupdate-xdi-qspi0-gold-rootfs";
|
||||||
static constexpr char CHIP_1_COPY_0_MOUNT_DIR[] = "/tmp/mntupdate-xdi-qspi1-nom-rootfs";
|
static constexpr char CHIP_1_COPY_0_MOUNT_DIR[] = "/tmp/mntupdate-xdi-qspi1-nom-rootfs";
|
||||||
static constexpr char CHIP_1_COPY_1_MOUNT_DIR[] = "/tmp/mntupdate-xdi-qspi1-gold-rootfs";
|
static constexpr char CHIP_1_COPY_1_MOUNT_DIR[] = "/tmp/mntupdate-xdi-qspi1-gold-rootfs";
|
||||||
|
static constexpr char LIST_DIR_DUMP_WORK_FILE[] = "/tmp/dir_listing.tmp";
|
||||||
|
|
||||||
static constexpr dur_millis_t INIT_SD_CARD_CHECK_TIMEOUT = 5000;
|
static constexpr dur_millis_t INIT_SD_CARD_CHECK_TIMEOUT = 5000;
|
||||||
static constexpr dur_millis_t DEFAULT_SD_CARD_CHECK_TIMEOUT = 60000;
|
static constexpr dur_millis_t DEFAULT_SD_CARD_CHECK_TIMEOUT = 60000;
|
||||||
@ -250,6 +251,12 @@ class CoreController : public ExtendedControllerBase, public ReceivesParameterMe
|
|||||||
|
|
||||||
ReturnValue_t actionListDirectoryIntoFile(ActionId_t actionId, MessageQueueId_t commandedBy,
|
ReturnValue_t actionListDirectoryIntoFile(ActionId_t actionId, MessageQueueId_t commandedBy,
|
||||||
const uint8_t* data, size_t size);
|
const uint8_t* data, size_t size);
|
||||||
|
ReturnValue_t actionListDirectoryDumpDirectly(ActionId_t actionId, MessageQueueId_t commandedBy,
|
||||||
|
const uint8_t* data, size_t size);
|
||||||
|
|
||||||
|
ReturnValue_t actionListDirectoryCommonCommandCreator(const uint8_t* data, size_t size,
|
||||||
|
std::ostringstream& oss);
|
||||||
|
|
||||||
ReturnValue_t actionXscReboot(const uint8_t* data, size_t size);
|
ReturnValue_t actionXscReboot(const uint8_t* data, size_t size);
|
||||||
ReturnValue_t actionReboot(const uint8_t* data, size_t size);
|
ReturnValue_t actionReboot(const uint8_t* data, size_t size);
|
||||||
|
|
||||||
|
@ -32,7 +32,6 @@ static constexpr char VERSION_FILE_NAME[] = "version.txt";
|
|||||||
static constexpr char REBOOT_FILE_NAME[] = "reboot.txt";
|
static constexpr char REBOOT_FILE_NAME[] = "reboot.txt";
|
||||||
static constexpr char TIME_FILE_NAME[] = "time_backup.txt";
|
static constexpr char TIME_FILE_NAME[] = "time_backup.txt";
|
||||||
|
|
||||||
static constexpr ActionId_t LIST_DIRECTORY_INTO_FILE = 0;
|
|
||||||
static constexpr ActionId_t ANNOUNCE_VERSION = 1;
|
static constexpr ActionId_t ANNOUNCE_VERSION = 1;
|
||||||
static constexpr ActionId_t ANNOUNCE_CURRENT_IMAGE = 2;
|
static constexpr ActionId_t ANNOUNCE_CURRENT_IMAGE = 2;
|
||||||
static constexpr ActionId_t ANNOUNCE_BOOT_COUNTS = 3;
|
static constexpr ActionId_t ANNOUNCE_BOOT_COUNTS = 3;
|
||||||
@ -59,6 +58,12 @@ static constexpr ActionId_t EXECUTE_SHELL_CMD_BLOCKING = 40;
|
|||||||
static constexpr ActionId_t EXECUTE_SHELL_CMD_NON_BLOCKING = 41;
|
static constexpr ActionId_t EXECUTE_SHELL_CMD_NON_BLOCKING = 41;
|
||||||
static constexpr ActionId_t SYSTEMCTL_CMD_EXECUTOR = 42;
|
static constexpr ActionId_t SYSTEMCTL_CMD_EXECUTOR = 42;
|
||||||
|
|
||||||
|
static constexpr ActionId_t LIST_DIRECTORY_INTO_FILE = 50;
|
||||||
|
static constexpr ActionId_t LIST_DIRECTORY_DUMP_DIRECTLY = 51;
|
||||||
|
static constexpr ActionId_t CP_HELPER = 52;
|
||||||
|
static constexpr ActionId_t MV_HELPER = 53;
|
||||||
|
static constexpr ActionId_t RM_HELPER = 54;
|
||||||
|
|
||||||
static constexpr uint8_t SUBSYSTEM_ID = SUBSYSTEM_ID::CORE;
|
static constexpr uint8_t SUBSYSTEM_ID = SUBSYSTEM_ID::CORE;
|
||||||
|
|
||||||
static constexpr Event ALLOC_FAILURE = event::makeEvent(SUBSYSTEM_ID, 0, severity::MEDIUM);
|
static constexpr Event ALLOC_FAILURE = event::makeEvent(SUBSYSTEM_ID, 0, severity::MEDIUM);
|
||||||
@ -96,6 +101,198 @@ static constexpr Event I2C_REBOOT = event::makeEvent(SUBSYSTEM_ID, 11, severity:
|
|||||||
//! [EXPORT] : [COMMENT] PDEC recovery through reset was not possible, performing full reboot.
|
//! [EXPORT] : [COMMENT] PDEC recovery through reset was not possible, performing full reboot.
|
||||||
static constexpr Event PDEC_REBOOT = event::makeEvent(SUBSYSTEM_ID, 12, severity::HIGH);
|
static constexpr Event PDEC_REBOOT = event::makeEvent(SUBSYSTEM_ID, 12, severity::HIGH);
|
||||||
|
|
||||||
|
class ListDirectoryCmdBase {
|
||||||
|
public: // TODO: Packet definition for clean deserialization
|
||||||
|
// 3 bytes for a and R flag, at least 5 bytes for minimum valid path /tmp with
|
||||||
|
// null termination
|
||||||
|
static constexpr size_t MIN_DATA_LEN = 8;
|
||||||
|
|
||||||
|
ListDirectoryCmdBase(const uint8_t* data, size_t maxSize) : data(data), maxSize(maxSize) {}
|
||||||
|
virtual ~ListDirectoryCmdBase() = default;
|
||||||
|
|
||||||
|
virtual ReturnValue_t parse() {
|
||||||
|
if (maxSize < MIN_DATA_LEN) {
|
||||||
|
return SerializeIF::STREAM_TOO_SHORT;
|
||||||
|
}
|
||||||
|
aFlag = data[0];
|
||||||
|
rFlag = data[1];
|
||||||
|
compressOption = data[2];
|
||||||
|
|
||||||
|
repoNameLen = strnlen(reinterpret_cast<const char*>(data + 3), maxSize - 3);
|
||||||
|
// Last byte MUST be null terminated!
|
||||||
|
if (repoNameLen >= maxSize - 3) {
|
||||||
|
return HasActionsIF::INVALID_PARAMETERS;
|
||||||
|
}
|
||||||
|
repoName = reinterpret_cast<const char*>(data + 3);
|
||||||
|
return returnvalue::OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool aFlagSet() const { return this->aFlag; }
|
||||||
|
bool rFlagSet() const { return this->rFlag; }
|
||||||
|
|
||||||
|
bool compressionOptionSet() const { return this->compressOption; }
|
||||||
|
|
||||||
|
const char* getRepoName(size_t& repoNameLen) const {
|
||||||
|
return this->repoName;
|
||||||
|
repoNameLen = this->repoNameLen;
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t getBaseSize() {
|
||||||
|
// Include NULL termination
|
||||||
|
if (repoName != nullptr) {
|
||||||
|
return 3 + repoNameLen + 1;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected:
|
||||||
|
const uint8_t* data;
|
||||||
|
size_t maxSize;
|
||||||
|
|
||||||
|
bool aFlag = false;
|
||||||
|
bool rFlag = false;
|
||||||
|
bool compressOption = false;
|
||||||
|
const char* repoName = nullptr;
|
||||||
|
size_t repoNameLen = 0;
|
||||||
|
};
|
||||||
|
|
||||||
|
class ListDirectoryIntoFile : public ListDirectoryCmdBase {
|
||||||
|
public:
|
||||||
|
// TODO: Packet definition for clean deserialization
|
||||||
|
// 3 bytes for a and R flag, at least 5 bytes for minimum valid path /tmp with
|
||||||
|
// null termination, at least 7 bytes for minimum target file name /tmp/a with
|
||||||
|
// null termination.
|
||||||
|
static constexpr size_t MIN_DATA_LEN = 15;
|
||||||
|
|
||||||
|
ListDirectoryIntoFile(const uint8_t* data, size_t maxSize)
|
||||||
|
: ListDirectoryCmdBase(data, maxSize) {}
|
||||||
|
ReturnValue_t parse() override {
|
||||||
|
if (maxSize < MIN_DATA_LEN) {
|
||||||
|
return SerializeIF::STREAM_TOO_SHORT;
|
||||||
|
}
|
||||||
|
ReturnValue_t result = ListDirectoryCmdBase::parse();
|
||||||
|
if (result != returnvalue::OK) {
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
targetNameLen =
|
||||||
|
strnlen(reinterpret_cast<const char*>(data + getBaseSize()), maxSize - getBaseSize());
|
||||||
|
if (targetNameLen >= maxSize - getBaseSize()) {
|
||||||
|
// Again: String MUST be null terminated.
|
||||||
|
return HasActionsIF::INVALID_PARAMETERS;
|
||||||
|
}
|
||||||
|
targetName = reinterpret_cast<const char*>(data + getBaseSize());
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
const char* getTargetName(size_t& targetNameLen) const {
|
||||||
|
return this->targetName;
|
||||||
|
targetNameLen = this->targetNameLen;
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
const char* targetName = nullptr;
|
||||||
|
size_t targetNameLen = 0;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct SourceTargetPair {
|
||||||
|
const char* sourceName = nullptr;
|
||||||
|
size_t sourceNameSize = 0;
|
||||||
|
const char* targetName = nullptr;
|
||||||
|
size_t targetNameSize = 0;
|
||||||
|
};
|
||||||
|
|
||||||
|
static ReturnValue_t parseDestTargetString(const uint8_t* data, size_t maxLen,
|
||||||
|
SourceTargetPair& destTgt) {
|
||||||
|
if (maxLen < 4) {
|
||||||
|
return SerializeIF::STREAM_TOO_SHORT;
|
||||||
|
}
|
||||||
|
destTgt.sourceNameSize = strnlen(reinterpret_cast<const char*>(data), maxLen);
|
||||||
|
if (destTgt.sourceNameSize >= maxLen) {
|
||||||
|
return HasActionsIF::INVALID_PARAMETERS;
|
||||||
|
}
|
||||||
|
destTgt.sourceName = reinterpret_cast<const char*>(data);
|
||||||
|
size_t remainingLen = maxLen - destTgt.sourceNameSize + 1;
|
||||||
|
if (remainingLen == 0) {
|
||||||
|
return HasActionsIF::INVALID_PARAMETERS;
|
||||||
|
}
|
||||||
|
destTgt.targetNameSize =
|
||||||
|
strnlen(reinterpret_cast<const char*>(data + destTgt.sourceNameSize + 1), remainingLen);
|
||||||
|
if (destTgt.targetNameSize >= remainingLen) {
|
||||||
|
return HasActionsIF::INVALID_PARAMETERS;
|
||||||
|
}
|
||||||
|
destTgt.targetName = reinterpret_cast<const char*>(data + destTgt.targetNameSize + 1);
|
||||||
|
return returnvalue::OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
class CpHelperParser {
|
||||||
|
public:
|
||||||
|
CpHelperParser(const uint8_t* data, size_t maxLen) : data(data), maxLen(maxLen) {}
|
||||||
|
|
||||||
|
ReturnValue_t parse() {
|
||||||
|
if (maxLen < 1) {
|
||||||
|
return SerializeIF::STREAM_TOO_SHORT;
|
||||||
|
}
|
||||||
|
recursiveOpt = data[0];
|
||||||
|
return parseDestTargetString(data + 1, maxLen - 1, destTgt);
|
||||||
|
}
|
||||||
|
const SourceTargetPair& destTgtPair() const { return destTgt; }
|
||||||
|
bool isRecursiveOptSet() const { return recursiveOpt; }
|
||||||
|
|
||||||
|
private:
|
||||||
|
const uint8_t* data;
|
||||||
|
size_t maxLen;
|
||||||
|
bool recursiveOpt = false;
|
||||||
|
SourceTargetPair destTgt;
|
||||||
|
};
|
||||||
|
|
||||||
|
class MvHelperParser {
|
||||||
|
public:
|
||||||
|
MvHelperParser(const uint8_t* data, size_t maxLen) : data(data), maxLen(maxLen) {}
|
||||||
|
|
||||||
|
ReturnValue_t parse() { return parseDestTargetString(data, maxLen, destTgt); }
|
||||||
|
const SourceTargetPair& destTgtPair() const { return destTgt; }
|
||||||
|
|
||||||
|
private:
|
||||||
|
const uint8_t* data;
|
||||||
|
size_t maxLen;
|
||||||
|
SourceTargetPair destTgt;
|
||||||
|
};
|
||||||
|
|
||||||
|
class RmHelperParser {
|
||||||
|
public:
|
||||||
|
RmHelperParser(const uint8_t* data, size_t maxLen) : data(data), maxLen(maxLen) {}
|
||||||
|
|
||||||
|
ReturnValue_t parse() {
|
||||||
|
if (maxLen < 2) {
|
||||||
|
return SerializeIF::STREAM_TOO_SHORT;
|
||||||
|
}
|
||||||
|
recursiveOpt = data[0];
|
||||||
|
forceOpt = data[1];
|
||||||
|
removeTargetSize = strnlen(reinterpret_cast<const char*>(data + 2), maxLen - 2);
|
||||||
|
// Must be null-terminated
|
||||||
|
if (removeTargetSize >= maxLen - 2) {
|
||||||
|
return HasActionsIF::INVALID_PARAMETERS;
|
||||||
|
}
|
||||||
|
removeTarget = reinterpret_cast<const char*>(data + 2);
|
||||||
|
return returnvalue::OK;
|
||||||
|
}
|
||||||
|
bool isRecursiveOptSet() const { return recursiveOpt; }
|
||||||
|
bool isForceOptSet() const { return forceOpt; }
|
||||||
|
|
||||||
|
const char* getRemoveTarget(size_t& removeTargetSize) {
|
||||||
|
removeTargetSize = this->removeTargetSize;
|
||||||
|
return removeTarget;
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
const uint8_t* data;
|
||||||
|
size_t maxLen;
|
||||||
|
bool recursiveOpt = false;
|
||||||
|
bool forceOpt = false;
|
||||||
|
const char* removeTarget = nullptr;
|
||||||
|
size_t removeTargetSize = 0;
|
||||||
|
};
|
||||||
|
|
||||||
} // namespace core
|
} // namespace core
|
||||||
|
|
||||||
#endif /* MISSION_SYSDEFS_H_ */
|
#endif /* MISSION_SYSDEFS_H_ */
|
||||||
|
Loading…
Reference in New Issue
Block a user