12#include <unordered_map>
17#include <tl/expected.hpp>
173uint16_t
getSlaveState(ecx_contextt* context, uint16_t slave);
276 const std::vector<uint16_t>& positions,
277 uint16_t requiredState, uint16_t targetState,
278 const char* targetName,
279 std::chrono::steady_clock::duration timeout,
280 std::chrono::steady_clock::duration resendInterval =
281 std::chrono::seconds(2),
282 std::function<
void()> tick =
nullptr,
283 std::function<
bool()> shouldAbort =
nullptr);
297bool setSlaveState(ecx_contextt* context, uint16_t slave, uint16_t targetState,
298 std::chrono::steady_clock::duration timeout =
299 std::chrono::seconds(5));
365 const std::vector<mm::comm::base::PdoMappingEntry>& entries);
378void logIoMap(uint8_t* ioMap,
size_t totalBytes);
473 supportedProtocols, writeLength, writeOffset)
482struct SlaveInfoSyncManager {
506 uint16_t startAddress;
531struct SlaveInfoFmmu {
545 uint8_t logicalEndBit;
553 uint8_t logicalStartBit;
560 uint32_t logicalStartAddress;
567 uint16_t logicalLength;
575 uint8_t physicalStartBit;
582 uint16_t physicalStartAddress;
594 logicalStartBit, logicalStartAddress,
595 logicalLength, physicalStartBit,
596 physicalStartAddress,
type)
611 uint16_t alStatusCode;
619 uint16_t aliasAddress;
626 std::vector<SlaveInfoFmmu> fmmus;
643 uint32_t productCode;
659 uint32_t revisionNumber;
667 std::vector<SlaveInfoSyncManager> syncManagers;
689 revisionNumber, syncManagers,
name, vendorId)
741 const
std::
string& iface);
827 bool setState(
int targetState,
int timeoutMs = 0,
int intervalMs = 100);
850 tl::expected<void, std::string>
loadParameters(
bool readValues =
false);
961 template <
typename T = std::vector<u
int8_t>>
964 std::lock_guard<std::mutex> lock(mailboxMutex_);
967 auto state = context_.slavelist[position_].state;
968 if (state == EC_STATE_INIT || state == EC_STATE_BOOT) {
969 throw std::runtime_error(
970 "To upload object dictionary entries, the device must be in the (2) "
971 "PRE-OPERATIONAL, (4) SAFE-OPERATIONAL, or (8) OPERATIONAL state. "
972 "The current state is " +
981 const std::unique_ptr<std::uint8_t[]> data =
982 std::make_unique<std::uint8_t[]>(parameter.byteLength);
983 auto wkc = ecx_SDOread(&context_, position_, parameter.index,
984 parameter.subindex,
false, ¶meter.byteLength,
987 LOG_F(ERROR,
"Device %d: SDO read failed for %#04x:%02x! WKC: %d",
988 position_, parameter.index, parameter.subindex, wkc);
991 throw std::runtime_error(
"Device " + std::to_string(position_) +
992 ": SDO read failed for " +
id +
993 "! WKC: " + std::to_string(wkc));
995 std::vector<uint8_t> receivedData(data.get(),
996 data.get() + parameter.byteLength);
997 if (!parameter.trySetValue(receivedData)) {
999 "Device %d: Failed to set the value of parameter %#04x:%02x!",
1000 position_, parameter.index, parameter.subindex);
1006 if constexpr (std::is_same_v<T, std::vector<uint8_t>>) {
1007 return parameter.data;
1010 auto value = parameter.getValue();
1011 if (
auto ptr = std::get_if<T>(&
value)) {
1014 throw std::bad_variant_access();
1043 template <
typename T = std::vector<u
int8_t>>
1045 std::lock_guard<std::mutex> lock(mailboxMutex_);
1048 auto state = context_.slavelist[position_].state;
1049 if (state == EC_STATE_INIT || state == EC_STATE_BOOT) {
1050 throw std::runtime_error(
1051 "To download object dictionary entries, the device must be in the "
1052 "(2) PRE-OPERATIONAL, (4) SAFE-OPERATIONAL, or (8) OPERATIONAL "
1053 "state. The current state is " +
1060 std::vector<uint8_t> data;
1061 if constexpr (std::is_same_v<T, std::vector<uint8_t>>) {
1065 if (!parameter.trySetValue(
value)) {
1067 "Device %d: Failed to set the value of parameter %#04x:%02x!",
1068 position_, parameter.index, parameter.subindex);
1071 data = parameter.data;
1075 if (data.size() != parameter.byteLength) {
1076 LOG_F(ERROR,
"Data size mismatch: expected %d, got %zu",
1077 parameter.byteLength, data.size());
1078 throw std::runtime_error(
"Data size does not match the parameter size!");
1082 auto wkc = ecx_SDOwrite(&context_, position_, parameter.index,
1083 parameter.subindex,
false, parameter.byteLength,
1088 LOG_F(ERROR,
"Device %d: SDO write failed for %#04x:%02x! WKC: %d",
1089 position_, parameter.index, parameter.subindex, wkc);
1114 void updatePdoMappings();
1127 void logPdoMappings()
const;
1158 void updateOutputs();
1181 void updateParametersFromInputs();
1208 tl::expected<std::vector<uint8_t>, std::string> readFile(
1209 const std::string& filename,
1210 size_t maxFileSize = 1 * 1024 * 1024,
1212 std::chrono::steady_clock::duration retryWait =
1213 std::chrono::milliseconds(100));
1228 tl::expected<std::vector<std::uint8_t>, std::string> readSdo(
1229 uint16_t
index, uint8_t subindex);
1244 tl::expected<ec_ODlistt, std::string> readODList(
1245 int retries = 5, std::chrono::steady_clock::duration retryWait =
1246 std::chrono::milliseconds(100));
1266 tl::expected<void, std::string> readODDescription(
1267 uint16_t item, ec_ODlistt& odList,
int retries = 5,
1268 std::chrono::steady_clock::duration retryWait =
1269 std::chrono::milliseconds(100));
1287 tl::expected<ec_OElistt, std::string> readOE(
1288 uint16_t item, ec_ODlistt& odList,
int retries = 5,
1289 std::chrono::steady_clock::duration retryWait =
1290 std::chrono::milliseconds(100));
1293 ecx_contextt& context_;
1294 const uint16_t position_;
1297 std::mutex mailboxMutex_;
1326 explicit Master(
const std::string& iface);
1379 void init(
const std::string& iface);
1507 const std::vector<std::unique_ptr<Slave>>&
slaves()
const;
1519 return fieldbus_->context.grouplist[fieldbus_->group].outputsWKC * 2 +
1520 fieldbus_->context.grouplist[fieldbus_->group].inputsWKC;
1590 std::unique_ptr<Fieldbus> fieldbus_;
1603 std::vector<std::unique_ptr<Slave>> slaves_;
Represents a device parameter identified by index and subindex.
Definition base.h:533
std::pair< uint16_t, uint8_t > Address
Alias for a pair of uint16_t and uint8_t representing a parameter address.
Definition base.h:543
int roundtrip()
Performs a complete send/receive process data cycle on the fieldbus.
Definition soem.cc:1416
Master()
Default constructor for the Master class.
Definition soem.cc:1337
const std::vector< std::unique_ptr< Slave > > & slaves() const
Returns a reference to the list of slaves.
Definition soem.cc:1412
tl::expected< int, std::string > initSlaves()
Initializes the slave devices on the EtherCAT network.
Definition soem.cc:1389
void exchangeProcessDataAndUpdateParameters()
Exchanges process data with slaves and updates parameters based on the response.
Definition soem.cc:1421
size_t getIoMapSize() const
Returns the total size of the I/O map for the current group.
Definition soem.cc:1384
Fieldbus & getFieldbus() const
Returns a reference to the Fieldbus instance.
Definition soem.cc:1380
void init(const std::string &iface)
Initializes the fieldbus using the specified network interface.
Definition soem.cc:1344
std::string getInterfaceName() const
Retrieves the interface name used by the fieldbus.
Definition soem.cc:1382
void deinit()
Deinitializes the fieldbus and releases associated resources.
Definition soem.cc:1362
bool start()
Initializes and starts the fieldbus master.
Definition soem.cc:1444
~Master()
Destructor for the Master class.
Definition soem.cc:1378
int expectedWkc() const
Calculates the expected working counter (WKC) for the current fieldbus group.
Definition soem.h:1518
uint32_t productCode() const
Gets the Product Code of the slave device.
Definition soem.cc:838
uint32_t vendorId() const
Gets the Vendor ID of the slave device.
Definition soem.cc:834
std::unordered_map< mm::comm::base::Parameter::Address, mm::comm::base::Parameter > & parametersMap()
Returns a reference to the internal parameters map.
Definition soem.cc:1131
mm::comm::base::Parameter & findParameter(uint16_t index, uint8_t subindex)
Finds and returns a reference to a Parameter in the parameters map.
Definition soem.cc:1135
uint32_t revisionNumber() const
Gets the Revision Number of the slave device.
Definition soem.cc:842
T upload(uint16_t index, uint8_t subindex, bool refresh=true)
Uploads a parameter value from the device's object dictionary.
Definition soem.h:962
tl::expected< void, std::string > loadParameters(bool readValues=false)
Loads the parameters (object dictionary entries) from the EtherCAT slave device.
Definition soem.cc:943
void clearParameters()
Clears all loaded object dictionary parameters.
Definition soem.cc:1127
void logTxPdoMappedParameters()
Logs all mapped TxPDO parameters with their current values.
Definition soem.cc:1112
Slave(ecx_contextt &context, uint16_t position)
Constructs a Slave object with a given context and position.
Definition soem.cc:829
void logRxPdoMappedParameters()
Logs all mapped RxPDO parameters with their current values.
Definition soem.cc:1097
bool setState(int targetState, int timeoutMs=0, int intervalMs=100)
Attempts to set the slave to the specified EtherCAT state and optionally waits until it is reached.
Definition soem.cc:848
void logParameters() const
Logs all loaded object dictionary parameters.
Definition soem.cc:1091
int getState() const
Reads the current state of the slave directly from the device.
Definition soem.cc:846
uint16_t position() const
Gets the position of the slave.
Definition soem.cc:832
bool download(uint16_t index, uint8_t subindex, const T &value)
Downloads a value to the specified object dictionary entry via SDO.
Definition soem.h:1044
uint8_t type
Definition co_dictionary.h:1
const char ** name
Definition co_dictionary.h:7
uint8_t * value
Definition co_dictionary.h:9
uint16_t index
Definition co_dictionary.h:0
std::string makeParameterId(int index, int subindex)
Formats the given index and subindex into a parameter identifier string.
Definition base.h:1654
mm::comm::base::PdoMappings readPdoMappings(ecx_contextt *context, uint16_t slave)
Read and parse all mapped PDO entries for a given slave.
Definition soem.cc:736
uint16_t getSlaveState(ecx_contextt *context, uint16_t slave)
Reads the current state of a specific EtherCAT slave directly from the device via an FPRD datagram.
Definition soem.cc:426
constexpr int kFoeTimeoutUs
Definition soem.h:29
void logPdoMappingEntries(const std::vector< mm::comm::base::PdoMappingEntry > &entries)
Logs PDO mapping entries grouped by PDO index.
Definition soem.cc:798
std::string convertSlaveStateToString(int state)
Converts a numeric EtherCAT state value into a human-readable string.
Definition soem.cc:441
constexpr int kSdoTimeoutUs
Definition soem.h:26
void transitionSlaves(ecx_contextt *context, const std::vector< uint16_t > &positions, uint16_t requiredState, uint16_t targetState, const char *targetName, std::chrono::steady_clock::duration timeout, std::chrono::steady_clock::duration resendInterval, std::function< void()> tick, std::function< bool()> shouldAbort)
Transitions multiple EtherCAT slaves to a target state using a single-threaded fire-then-poll pattern...
Definition soem.cc:470
std::string uiConfigWithDefaultPdoMapping
Default UI configuration used for PDO mapping.
Definition soem.h:59
void initFieldbus(std::unique_ptr< Fieldbus > &fieldbus, const std::string &iface)
Initializes a Fieldbus instance for EtherCAT communication.
Definition soem.cc:1304
void updateMailboxSyncManagersOnNextState(ecx_contextt *context, uint16_t slave, int targetState)
Updates the mailbox and sync manager (SM) configurations for an EtherCAT slave when transitioning to ...
Definition soem.cc:571
void logIoMap(uint8_t *ioMap, size_t totalBytes)
Logs the I/O map as a formatted hexadecimal string.
Definition soem.cc:819
void checkPdoMapping(ecx_contextt *context, uint16 slave)
Diagnoses and prints the PDO mapping for a given slave.
Definition soem.cc:344
void configureDetectedSmmModule(ecx_contextt *context, uint16_t slave)
Reads the detected SMM module ID from the slave device and configures the module identification objec...
Definition soem.cc:663
std::map< std::string, std::string > mapMacAddressesToInterfaces()
Maps MAC addresses to their corresponding network interface names.
Definition soem.cc:42
int setPdoMappingFromUiConfig(ecx_contextt *context, uint16 slave)
Sets PDO mapping for a specified EtherCAT slave using UI configuration.
Definition soem.cc:221
uint16 position
Definition soem.h:703
void logSlaveInfo(ecx_contextt *context, uint16 position)
Logs detailed information about a specific EtherCAT slave in JSON format.
Definition soem.cc:215
NLOHMANN_DEFINE_TYPE_NON_INTRUSIVE(SlaveInfoMailbox, readLength, readOffset, supportedProtocols, writeLength, writeOffset) struct SlaveInfoSyncManager
Information about a slave's Sync Manager (SM) configuration.
Definition soem.h:472
SlaveInfo getSlaveInfo(ecx_contextt *context, uint16 position)
Definition soem.cc:163
std::vector< uint8_t > readSii(ecx_contextt *context, uint16 position)
Read the complete Slave Information Interface (SII) EEPROM from an EtherCAT slave device.
Definition soem.cc:693
bool setSlaveState(ecx_contextt *context, uint16_t slave, uint16_t targetState, std::chrono::steady_clock::duration timeout)
Transitions a single EtherCAT slave to targetState.
Definition soem.cc:563
Specialization of std::hash for std::pair<uint16_t, uint8_t>.
Definition util.h:612
Represents the mapped PDO entries for a slave device.
Definition base.h:175
Structure representing the Fieldbus configuration and context for EtherCAT communication.
Definition soem.h:394
boolean ecaterror
Definition soem.h:410
ec_idxstackT idxstack
Definition soem.h:409
ec_eepromFMMUt eepFMMU
Definition soem.h:416
ec_groupt grouplist[EC_MAXGROUP]
Definition soem.h:405
std::string iface
Definition soem.h:396
uint8 group
Definition soem.h:397
int slavecount
Definition soem.h:404
uint8 esibuf[EC_MAXEEPBUF]
Definition soem.h:406
uint32 esimap[EC_MAXEEPBITMAP]
Definition soem.h:407
int64 DCtime
Definition soem.h:411
ec_eepromSMt eepSM
Definition soem.h:415
ec_PDOassignt PDOassign[EC_MAX_MAPT]
Definition soem.h:413
uint8 map[4096]
Definition soem.h:401
ecx_contextt context
Definition soem.h:395
int roundtripTime
Definition soem.h:398
ec_PDOdesct PDOdesc[EC_MAX_MAPT]
Definition soem.h:414
ecx_portt port
Definition soem.h:402
ec_slavet slavelist[EC_MAXSLAVE]
Definition soem.h:403
ec_SMcommtypet SMcommtype[EC_MAX_MAPT]
Definition soem.h:412
ec_eringt elist
Definition soem.h:408
Information about a slave's mailbox configuration.
Definition soem.h:426
uint16_t writeOffset
Offset of the mailbox write area within the slave's memory.
Definition soem.h:469
uint16_t supportedProtocols
Supported mailbox protocols as a bitmask.
Definition soem.h:453
uint16_t readOffset
Offset of the mailbox read area within the slave's memory.
Definition soem.h:441
uint16_t readLength
Length of the mailbox read area (in bytes).
Definition soem.h:433
uint16_t writeLength
Length of the mailbox write area (in bytes).
Definition soem.h:461