/***************************************************************************** * * Copyright (C) 2006-2024 Florian Pose, Ingenieurgemeinschaft IgH * * This file is part of the IgH EtherCAT Master. * * The IgH EtherCAT Master is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License version 2, as * published by the Free Software Foundation. * * The IgH EtherCAT Master is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General * Public License for more details. * * You should have received a copy of the GNU General Public License along * with the IgH EtherCAT Master; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA * ****************************************************************************/ #include #include #include using namespace std; #include "CommandFoeRead.h" #include "foe.h" #include "MasterDevice.h" /****************************************************************************/ CommandFoeRead::CommandFoeRead(): FoeCommand("foe_read", "Read a file from a slave via FoE.") { } /****************************************************************************/ string CommandFoeRead::helpString(const string &binaryBaseName) const { stringstream str; str << binaryBaseName << " " << getName() << " [OPTIONS] " << endl << endl << getBriefDescription() << endl << endl << "This command requires a single slave to be selected." << endl << endl << "Arguments:" << endl << " SOURCEFILE is the name of the source file on the slave." << endl << endl << "Command-specific options:" << endl << " --output-file -o Local target filename. If" << endl << " '-' (default), data are" << endl << " printed to stdout." << endl << " --alias -a " << endl << " --position -p Slave selection. See the help" << endl << " of the 'slaves' command." << endl << endl << numericInfo(); return str.str(); } /****************************************************************************/ void CommandFoeRead::execute(const StringVector &args) { SlaveList slaves; ec_ioctl_slave_t *slave; ec_ioctl_slave_foe_t data; unsigned int i; stringstream err; if (args.size() != 1) { err << "'" << getName() << "' takes exactly one argument!"; throwInvalidUsageException(err); } MasterDevice m(getSingleMasterIndex()); m.open(MasterDevice::ReadWrite); slaves = selectedSlaves(m); if (slaves.size() != 1) { throwSingleSlaveRequired(slaves.size()); } slave = &slaves.front(); data.slave_position = slave->position; /* FIXME: No good idea to have a fixed buffer size. * Read in chunks and fill a buffer instead. */ data.offset = 0; data.buffer_size = 0x8800; data.buffer = new uint8_t[data.buffer_size]; strncpy(data.file_name, args[0].c_str(), sizeof(data.file_name) - 1); try { m.readFoe(&data); } catch (MasterDeviceException &e) { delete [] data.buffer; if (data.result) { if (data.result == FOE_OPCODE_ERROR) { err << "FoE read aborted with error code 0x" << setw(8) << setfill('0') << hex << data.error_code << ": " << errorText(data.error_code); } else { err << "Failed to write via FoE: " << resultText(data.result); } throwCommandException(err); } else { throw e; } } // TODO --output-file for (i = 0; i < data.data_size; i++) { uint8_t *w = data.buffer + i; cout << *(uint8_t *) w ; } delete [] data.buffer; } /****************************************************************************/