diff --git a/Makefile b/Makefile new file mode 100644 index 00000000..fc5dd8ee --- /dev/null +++ b/Makefile @@ -0,0 +1,9 @@ +all: + $(MAKE) -C src all + +clean: + $(MAKE) -C src realclean + +install: + $(MAKE) -C src install-rt install-user + diff --git a/TODO b/TODO new file mode 100644 index 00000000..0a19d879 --- /dev/null +++ b/TODO @@ -0,0 +1,5 @@ +- (Re-)testing/Documentation of all devices +- option --enable-ethercat for configure script +- DOC: don't use EL9010 +- Debug EL5151/EL5152/EL2521 Subindex does not exist +- DOC: Parameter A60/A61 Stoeber diff --git a/debian/changelog b/debian/changelog new file mode 100644 index 00000000..6007f62a --- /dev/null +++ b/debian/changelog @@ -0,0 +1,5 @@ +linuxcnc-ethercat (0.9.0-1) unstable; urgency=low + + * Initial release + + -- Sascha Ittner Mon, 12 Jan 2015 10:19:04 +0100 diff --git a/debian/compat b/debian/compat new file mode 100644 index 00000000..45a4fb75 --- /dev/null +++ b/debian/compat @@ -0,0 +1 @@ +8 diff --git a/debian/control b/debian/control new file mode 100644 index 00000000..8a627bec --- /dev/null +++ b/debian/control @@ -0,0 +1,11 @@ +Source: linuxcnc-ethercat +Section: unknown +Priority: extra +Maintainer: linuxcnc +Build-Depends: debhelper (>= 8.0.0), linuxcnc-dev, etherlabmaster-dev +Standards-Version: 3.9.3 + +Package: linuxcnc-ethercat +Architecture: any +Depends: ${shlibs:Depends}, ${misc:Depends}, linuxcnc, etherlabmaster +Description: LinuxCNC EtherCAT HAL driver diff --git a/debian/copyright b/debian/copyright new file mode 100644 index 00000000..53b4dfc3 --- /dev/null +++ b/debian/copyright @@ -0,0 +1,33 @@ +Format: http://www.debian.org/doc/packaging-manuals/copyright-format/1.0/ +Upstream-Name: linuxcnc-ethercat +Source: + +Files: * +Copyright: 2015 Sascha Ittner +License: GPL-2.0+ + +Files: debian/* +Copyright: 2015 Sascha Ittner +License: GPL-2.0+ + +License: GPL-2.0+ + This package is free software; you can redistribute it and/or modify + it under the terms of the GNU General Public License as published by + the Free Software Foundation; either version 2 of the License, or + (at your option) any later version. + . + This package 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 this program. If not, see + . + On Debian systems, the complete text of the GNU General + Public License version 2 can be found in "/usr/share/common-licenses/GPL-2". + +# Please also look if there are files or directories which have a +# different copyright/license attached and list them here. +# Please avoid to pick license terms that are more restrictive than the +# packaged work, as it may make Debian's contributions unacceptable upstream. diff --git a/debian/docs b/debian/docs new file mode 100644 index 00000000..1333ed77 --- /dev/null +++ b/debian/docs @@ -0,0 +1 @@ +TODO diff --git a/debian/rules b/debian/rules new file mode 100755 index 00000000..b760bee7 --- /dev/null +++ b/debian/rules @@ -0,0 +1,13 @@ +#!/usr/bin/make -f +# -*- makefile -*- +# Sample debian/rules that uses debhelper. +# This file was originally written by Joey Hess and Craig Small. +# As a special exception, when this file is copied by dh-make into a +# dh-make output file, you may use that output file without restriction. +# This special exception was added by Craig Small in version 0.37 of dh-make. + +# Uncomment this to turn on verbose mode. +#export DH_VERBOSE=1 + +%: + dh $@ diff --git a/debian/source/format b/debian/source/format new file mode 100644 index 00000000..d3827e75 --- /dev/null +++ b/debian/source/format @@ -0,0 +1 @@ +1.0 diff --git a/src/Makefile b/src/Makefile new file mode 100644 index 00000000..2a9043b0 --- /dev/null +++ b/src/Makefile @@ -0,0 +1,49 @@ +MODINC=$(shell comp --print-modinc) + +all: rt user + +realclean: + rm -f rm *.ko *.mod.c *.o .*.cmd + rm -f modules.order Module.symvers + rm -rf .tmp_versions + rm -f lcec_conf + +obj-m += lcec.o +lcec-objs := \ + lcec_main.o \ + lcec_generic.o \ + lcec_el1xxx.o \ + lcec_el2521.o \ + lcec_el2xxx.o \ + lcec_el31x2.o \ + lcec_el40x1.o \ + lcec_el40x2.o \ + lcec_el41x2.o \ + lcec_el5101.o \ + lcec_el5151.o \ + lcec_el5152.o \ + lcec_el7342.o \ + lcec_el95xx.o \ + lcec_em7004.o \ + lcec_stmds5k.o \ + lcec_deasda.o \ + $(MATHSTUB) + +include $(MODINC) + +user: lcec_conf + +lcec_conf: lcec_conf.c + gcc $(EXTRA_CFLAGS) -Wframe-larger-than=16384 -URTAPI -U__MODULE__ -DULAPI -Os -o $@ $< -Wl,-rpath,$(LIBDIR) -L$(LIBDIR) -llinuxcnchal -lexpat + +install-user: + mkdir -p $(DESTDIR)$(EMC2_HOME)/bin + cp lcec_conf $(DESTDIR)$(EMC2_HOME)/bin + +rt: + $(MAKE) KBUILD_EXTRA_SYMBOLS="$(RTLIBDIR)/Module.symvers $(RTLIBDIR)/../ethercat/Module.symvers" -C $(KERNELDIR) SUBDIRS=`pwd` CC=$(CC) V=0 modules + +install-rt: + mkdir -p $(DESTDIR)$(RTLIBDIR) + $(MAKE) install + diff --git a/src/lcec.h b/src/lcec.h new file mode 100644 index 00000000..8b9192e2 --- /dev/null +++ b/src/lcec.h @@ -0,0 +1,165 @@ +// +// Copyright (C) 2011 Sascha Ittner +// +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2 of the License, or +// (at your option) any later version. +// +// This program 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 this program; if not, write to the Free Software +// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA +// +#ifndef _LCEC_H_ +#define _LCEC_H_ + +#include "lcnc_rtapi.h" + +#include "hal.h" + +#include "rtapi.h" +#include "rtapi_ctype.h" +#include "rtapi_app.h" +#include "rtapi_string.h" +#include "rtapi_math.h" + +#include "ecrt.h" +#include "lcec_conf.h" + +// list macros +#define LCEC_LIST_APPEND(first, last, item) \ +do { \ + (item)->prev = (last); \ + if ((item)->prev != NULL) { \ + (item)->prev->next = (item); \ + } else { \ + (first) = (item); \ + } \ + (last) = (item); \ +} while (0); \ + +// pdo macros +#define LCEC_PDO_INIT(pdo, pos, vid, pid, idx, sidx, off, bpos) \ +do { \ + pdo->position = pos; \ + pdo->vendor_id = vid; \ + pdo->product_code = pid; \ + pdo->index = idx; \ + pdo->subindex = sidx; \ + pdo->offset = off; \ + pdo->bit_position = bpos; \ + pdo++; \ +} while (0); \ + +#define LCEC_MSG_PFX "LCEC: " + +// vendor ids +#define LCEC_BECKHOFF_VID 0x00000002 +#define LCEC_STOEBER_VID 0x000000b9 +#define LCEC_DELTA_VID 0x000001dd + +// SDO request timeout (ms) +#define LCEC_SDO_REQ_TIMEOUT 1000 + +struct lcec_master; +struct lcec_slave; + +typedef int (*lcec_slave_init_t) (int comp_id, struct lcec_slave *slave, ec_pdo_entry_reg_t *pdo_entry_regs); +typedef void (*lcec_slave_cleanup_t) (struct lcec_slave *slave); +typedef void (*lcec_slave_rw_t) (struct lcec_slave *slave, long period); + +typedef struct lcec_master_data { + hal_u32_t *slaves_responding; + hal_bit_t *state_init; + hal_bit_t *state_preop; + hal_bit_t *state_safeop; + hal_bit_t *state_op; + hal_bit_t *link_up; + hal_bit_t *all_op; +} lcec_master_data_t; + +typedef struct lcec_slave_state { + hal_bit_t *online; + hal_bit_t *operational; + hal_bit_t *state_init; + hal_bit_t *state_preop; + hal_bit_t *state_safeop; + hal_bit_t *state_op; +} lcec_slave_state_t; + +typedef struct lcec_master { + struct lcec_master *prev; + struct lcec_master *next; + int index; + char name[LCEC_CONF_STR_MAXLEN]; + ec_master_t *master; + unsigned long mutex; + int pdo_entry_count; + ec_pdo_entry_reg_t *pdo_entry_regs; + ec_domain_t *domain; + uint8_t *process_data; + int process_data_len; + struct lcec_slave *first_slave; + struct lcec_slave *last_slave; + lcec_master_data_t *hal_data; + uint64_t app_time; + uint32_t app_time_period; + int sync_ref_cnt; + int sync_ref_cycles; +} lcec_master_t; + +typedef struct { + uint16_t assignActivate; + uint32_t sync0Cycle; + uint32_t sync0Shift; + uint32_t sync1Cycle; + uint32_t sync1Shift; +} lcec_slave_dc_t; + +typedef struct { + uint16_t divider; + uint16_t intervals; +} lcec_slave_watchdog_t; + +typedef struct { + uint16_t index; + int16_t subindex; + size_t length; + uint8_t data[]; +} lcec_slave_sdoconf_t; + +typedef struct lcec_slave { + struct lcec_slave *prev; + struct lcec_slave *next; + struct lcec_master *master; + int index; + char name[LCEC_CONF_STR_MAXLEN]; + uint32_t vid; + uint32_t pid; + int pdo_entry_count; + ec_sync_info_t *sync_info; + ec_slave_config_t *config; + ec_slave_config_state_t state; + lcec_slave_dc_t *dc_conf; + lcec_slave_watchdog_t *wd_conf; + lcec_slave_init_t proc_init; + lcec_slave_cleanup_t proc_cleanup; + lcec_slave_rw_t proc_read; + lcec_slave_rw_t proc_write; + lcec_slave_state_t *hal_state_data; + void *hal_data; + ec_pdo_entry_info_t *generic_pdo_entries; + ec_pdo_info_t *generic_pdos; + ec_sync_info_t *generic_sync_managers; + lcec_slave_sdoconf_t *sdo_config; +} lcec_slave_t; + +ec_sdo_request_t *lcec_read_sdo(struct lcec_slave *slave, uint16_t index, uint8_t subindex, size_t size); + +#endif + diff --git a/src/lcec_conf.c b/src/lcec_conf.c new file mode 100644 index 00000000..18bbc76c --- /dev/null +++ b/src/lcec_conf.c @@ -0,0 +1,1124 @@ +// +// Copyright (C) 2012 Sascha Ittner +// +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2 of the License, or +// (at your option) any later version. +// +// This program 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 this program; if not, write to the Free Software +// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA +// + +#include +#include +#include +#include +#include +#include + +#include "rtapi.h" +#include "hal.h" + +#include "lcec_conf.h" + +#define BUFFSIZE 8192 + +typedef struct { + char *name; + LCEC_SLAVE_TYPE_T type; +} LCEC_CONF_TYPELIST_T; + +typedef struct { + hal_u32_t *master_count; + hal_u32_t *slave_count; +} LCEC_CONF_HAL_T; + +static const LCEC_CONF_TYPELIST_T slaveTypes[] = { + // bus coupler + { "EK1100", lcecSlaveTypeEK1100 }, + + // generic device + { "generic", lcecSlaveTypeGeneric }, + + // digital in + { "EL1002", lcecSlaveTypeEL1002 }, + { "EL1004", lcecSlaveTypeEL1004 }, + { "EL1008", lcecSlaveTypeEL1008 }, + { "EL1012", lcecSlaveTypeEL1012 }, + { "EL1014", lcecSlaveTypeEL1014 }, + { "EL1018", lcecSlaveTypeEL1018 }, + { "EL1024", lcecSlaveTypeEL1024 }, + { "EL1034", lcecSlaveTypeEL1034 }, + { "EL1084", lcecSlaveTypeEL1084 }, + { "EL1088", lcecSlaveTypeEL1088 }, + { "EL1094", lcecSlaveTypeEL1094 }, + { "EL1098", lcecSlaveTypeEL1098 }, + { "EL1104", lcecSlaveTypeEL1104 }, + { "EL1114", lcecSlaveTypeEL1114 }, + { "EL1124", lcecSlaveTypeEL1124 }, + { "EL1134", lcecSlaveTypeEL1134 }, + { "EL1144", lcecSlaveTypeEL1144 }, + { "EL1808", lcecSlaveTypeEL1808 }, + { "EL1809", lcecSlaveTypeEL1809 }, + + // digital out + { "EL2002", lcecSlaveTypeEL2002 }, + { "EL2004", lcecSlaveTypeEL2004 }, + { "EL2008", lcecSlaveTypeEL2008 }, + { "EL2022", lcecSlaveTypeEL2022 }, + { "EL2024", lcecSlaveTypeEL2024 }, + { "EL2032", lcecSlaveTypeEL2032 }, + { "EL2034", lcecSlaveTypeEL2034 }, + { "EL2042", lcecSlaveTypeEL2042 }, + { "EL2084", lcecSlaveTypeEL2084 }, + { "EL2088", lcecSlaveTypeEL2088 }, + { "EL2124", lcecSlaveTypeEL2124 }, + { "EL2808", lcecSlaveTypeEL2808 }, + { "EL2809", lcecSlaveTypeEL2809 }, + + // analog in, 2ch, 16 bits + { "EL3102", lcecSlaveTypeEL3102 }, + { "EL3112", lcecSlaveTypeEL3112 }, + { "EL3122", lcecSlaveTypeEL3122 }, + { "EL3142", lcecSlaveTypeEL3142 }, + { "EL3152", lcecSlaveTypeEL3152 }, + { "EL3162", lcecSlaveTypeEL3162 }, + + // analog out, 1ch, 12 bits + { "EL4001", lcecSlaveTypeEL4001 }, + { "EL4011", lcecSlaveTypeEL4011 }, + { "EL4021", lcecSlaveTypeEL4021 }, + { "EL4031", lcecSlaveTypeEL4031 }, + + // analog out, 2ch, 12 bits + { "EL4002", lcecSlaveTypeEL4002 }, + { "EL4012", lcecSlaveTypeEL4012 }, + { "EL4022", lcecSlaveTypeEL4022 }, + { "EL4032", lcecSlaveTypeEL4032 }, + + // analog out, 2ch, 16 bits + { "EL4102", lcecSlaveTypeEL4102 }, + { "EL4112", lcecSlaveTypeEL4112 }, + { "EL4122", lcecSlaveTypeEL4122 }, + { "EL4132", lcecSlaveTypeEL4132 }, + + // encoder inputs + { "EL5101", lcecSlaveTypeEL5101 }, + { "EL5151", lcecSlaveTypeEL5151 }, + { "EL5152", lcecSlaveTypeEL5152 }, + + // pulse train (stepper) output + { "EL2521", lcecSlaveTypeEL2521 }, + + // dc servo + { "EL7342", lcecSlaveTypeEL7342 }, + + // power suppply + { "EL9505", lcecSlaveTypeEL9505 }, + { "EL9508", lcecSlaveTypeEL9508 }, + { "EL9510", lcecSlaveTypeEL9510 }, + { "EL9512", lcecSlaveTypeEL9512 }, + { "EL9515", lcecSlaveTypeEL9515 }, + + // multi axis interface + { "EM7004", lcecSlaveTypeEM7004 }, + + // stoeber MDS5000 series + { "StMDS5k", lcecSlaveTypeStMDS5k }, + + // Delta ASDA series + { "DeASDA", lcecSlaveTypeDeASDA }, + + { NULL } +}; + +char *modname = "lcec_conf"; +int hal_comp_id; +LCEC_CONF_HAL_T *conf_hal_data; + +int exitEvent; + +XML_Parser parser; +int currConfType; +void *outputBuffer; +size_t outputBufferLen; +size_t outputBufferPos; + +LCEC_CONF_MASTER_T *currMaster; +LCEC_CONF_SLAVE_T *currSlave; +LCEC_CONF_SYNCMANAGER_T *currSyncManager; +LCEC_CONF_PDO_T *currPdo; +LCEC_CONF_SDOCONF_T *currSdoConf; + +int shmem_id; + +void xml_start_handler(void *data, const char *el, const char **attr); +void xml_end_handler(void *data, const char *el); + +void *getOutputBuffer(size_t len); + +int parseHexdump(const char *str, uint8_t *buf); + +void parseMasterAttrs(const char **attr); +void parseSlaveAttrs(const char **attr); +void parseDcConfAttrs(const char **attr); +void parseWatchdogAttrs(const char **attr); +void parseSdoConfigAttrs(const char **attr); +void parseSdoDataRawAttrs(const char **attr); +void parseSyncManagerAttrs(const char **attr); +void parsePdoAttrs(const char **attr); +void parsePdoEntryAttrs(const char **attr); + +int parseSyncCycle(const char *nptr); + +static void exitHandler(int sig) { + uint64_t u = 1; + if (write(exitEvent, &u, sizeof(uint64_t)) < 0) { + fprintf(stderr, "%s: ERROR: error writing exit event\n", modname); + } +} + +int main(int argc, char **argv) { + int ret = 1; + char *filename; + int done; + char buffer[BUFFSIZE]; + FILE *file; + LCEC_CONF_NULL_T *end; + void *shmem_ptr; + LCEC_CONF_HEADER_T *header; + uint64_t u; + + // initialize component + hal_comp_id = hal_init(modname); + if (hal_comp_id < 1) { + fprintf(stderr, "%s: ERROR: hal_init failed\n", modname); + goto fail0; + } + + // allocate hal memory + conf_hal_data = hal_malloc(sizeof(LCEC_CONF_HAL_T)); + if (conf_hal_data == NULL) { + fprintf(stderr, "%s: ERROR: unable to allocate HAL shared memory\n", modname); + goto fail1; + } + + // register pins + if (hal_pin_u32_newf(HAL_OUT, &(conf_hal_data->master_count), hal_comp_id, "%s.conf.master-count", LCEC_MODULE_NAME) != 0) { + fprintf(stderr, "%s: ERROR: unable to register pin %s.conf.master-count\n", modname, LCEC_MODULE_NAME); + goto fail1; + } + if (hal_pin_u32_newf(HAL_OUT, &(conf_hal_data->slave_count), hal_comp_id, "%s.conf.slave-count", LCEC_MODULE_NAME) != 0) { + fprintf(stderr, "%s: ERROR: unable to register pin %s.conf.slave-count\n", modname, LCEC_MODULE_NAME); + goto fail1; + } + *conf_hal_data->master_count = 0; + *conf_hal_data->slave_count = 0; + + // initialize signal handling + exitEvent = eventfd(0, 0); + if (exitEvent == -1) { + fprintf(stderr, "%s: ERROR: unable to create exit event\n", modname); + goto fail1; + } + signal(SIGINT, exitHandler); + signal(SIGTERM, exitHandler); + + // get config file name + if (argc != 2) { + fprintf(stderr, "%s: ERROR: invalid arguments\n", modname); + goto fail2; + } + filename = argv[1]; + + // open file + file = fopen(filename, "r"); + if (file == NULL) { + fprintf(stderr, "%s: ERROR: unable to open config file %s\n", modname, filename); + goto fail2; + } + + // create xml parser + parser = XML_ParserCreate(NULL); + if (parser == NULL) { + fprintf(stderr, "%s: ERROR: Couldn't allocate memory for parser\n", modname); + goto fail3; + } + + // setup handlers + XML_SetElementHandler(parser, xml_start_handler, xml_end_handler); + + currConfType = lcecConfTypeNone; + outputBuffer = NULL; + outputBufferLen = 0; + outputBufferPos = 0; + currMaster = NULL; + currSlave = NULL; + currSyncManager = NULL; + currPdo = NULL; + currSdoConf = NULL; + for (done=0; !done;) { + // read block + int len = fread(buffer, 1, BUFFSIZE, file); + if (ferror(file)) { + fprintf(stderr, "%s: ERROR: Couldn't read from file %s\n", modname, filename); + goto fail4; + } + + // check for EOF + done = feof(file); + + // parse current block + if (!XML_Parse(parser, buffer, len, done)) { + fprintf(stderr, "%s: ERROR: Parse error at line %u: %s\n", modname, + (unsigned int)XML_GetCurrentLineNumber(parser), + XML_ErrorString(XML_GetErrorCode(parser))); + goto fail4; + } + } + + // set end marker + end = getOutputBuffer(sizeof(LCEC_CONF_NULL_T)); + if (end == NULL) { + goto fail4; + } + end->confType = lcecConfTypeNone; + + // setup shared mem for config + shmem_id = rtapi_shmem_new(LCEC_CONF_SHMEM_KEY, hal_comp_id, sizeof(LCEC_CONF_HEADER_T) + outputBufferPos); + if ( shmem_id < 0 ) { + fprintf(stderr, "%s: ERROR: couldn't allocate user/RT shared memory\n", modname); + goto fail4; + } + if (rtapi_shmem_getptr(shmem_id, &shmem_ptr) < 0) { + fprintf(stderr, "%s: ERROR: couldn't map user/RT shared memory\n", modname); + goto fail5; + } + + // setup header + header = shmem_ptr; + shmem_ptr += sizeof(LCEC_CONF_HEADER_T); + header->magic = LCEC_CONF_SHMEM_MAGIC; + header->length = outputBufferPos; + + // copy data + memcpy(shmem_ptr, outputBuffer, outputBufferPos); + + // free build buffer + free(outputBuffer); + outputBuffer = NULL; + + // everything is fine + ret = 0; + hal_ready(hal_comp_id); + + // wait for SIGTERM + if (read(exitEvent, &u, sizeof(uint64_t)) < 0) { + fprintf(stderr, "%s: ERROR: error reading exit event\n", modname); + } + +fail5: + rtapi_shmem_delete(shmem_id, hal_comp_id); +fail4: + if (outputBuffer != NULL) { + free(outputBuffer); + } + XML_ParserFree(parser); +fail3: + fclose(file); +fail2: + close(exitEvent); +fail1: + hal_exit(hal_comp_id); +fail0: + return ret; +} + +void xml_start_handler(void *data, const char *el, const char **attr) { + switch (currConfType) { + case lcecConfTypeNone: + if (strcmp(el, "masters") == 0) { + currConfType = lcecConfTypeMasters; + return; + } + break; + case lcecConfTypeMasters: + if (strcmp(el, "master") == 0) { + currConfType = lcecConfTypeMaster; + parseMasterAttrs(attr); + return; + } + break; + case lcecConfTypeMaster: + if (strcmp(el, "slave") == 0) { + currConfType = lcecConfTypeSlave; + parseSlaveAttrs(attr); + return; + } + break; + case lcecConfTypeSlave: + if (strcmp(el, "dcConf") == 0) { + currConfType = lcecConfTypeDcConf; + parseDcConfAttrs(attr); + return; + } + if (strcmp(el, "watchdog") == 0) { + currConfType = lcecConfTypeWatchdog; + parseWatchdogAttrs(attr); + return; + } + if (strcmp(el, "sdoConfig") == 0) { + currConfType = lcecConfTypeSdoConfig; + parseSdoConfigAttrs(attr); + return; + } + if (currSlave->type == lcecSlaveTypeGeneric && strcmp(el, "syncManager") == 0) { + currConfType = lcecConfTypeSyncManager; + parseSyncManagerAttrs(attr); + return; + } + break; + case lcecConfTypeSdoConfig: + if (strcmp(el, "sdoDataRaw") == 0) { + currConfType = lcecConfTypeSdoDataRaw; + parseSdoDataRawAttrs(attr); + return; + } + break; + case lcecConfTypeSyncManager: + if (strcmp(el, "pdo") == 0) { + currConfType = lcecConfTypePdo; + parsePdoAttrs(attr); + return; + } + break; + case lcecConfTypePdo: + if (strcmp(el, "pdoEntry") == 0) { + currConfType = lcecConfTypePdoEntry; + parsePdoEntryAttrs(attr); + return; + } + break; + } + + fprintf(stderr, "%s: ERROR: unexpected node %s found\n", modname, el); + XML_StopParser(parser, 0); +} + +void xml_end_handler(void *data, const char *el) { + switch (currConfType) { + case lcecConfTypeMasters: + if (strcmp(el, "masters") == 0) { + currConfType = lcecConfTypeNone; + return; + } + break; + case lcecConfTypeMaster: + if (strcmp(el, "master") == 0) { + currConfType = lcecConfTypeMasters; + return; + } + break; + case lcecConfTypeSlave: + if (strcmp(el, "slave") == 0) { + currConfType = lcecConfTypeMaster; + return; + } + break; + case lcecConfTypeDcConf: + if (strcmp(el, "dcConf") == 0) { + currConfType = lcecConfTypeSlave; + return; + } + break; + case lcecConfTypeWatchdog: + if (strcmp(el, "watchdog") == 0) { + currConfType = lcecConfTypeSlave; + return; + } + break; + case lcecConfTypeSdoConfig: + if (strcmp(el, "sdoConfig") == 0) { + currConfType = lcecConfTypeSlave; + return; + } + break; + case lcecConfTypeSdoDataRaw: + if (strcmp(el, "sdoDataRaw") == 0) { + currConfType = lcecConfTypeSdoConfig; + return; + } + break; + case lcecConfTypeSyncManager: + if (strcmp(el, "syncManager") == 0) { + currConfType = lcecConfTypeSlave; + return; + } + break; + case lcecConfTypePdo: + if (strcmp(el, "pdo") == 0) { + currConfType = lcecConfTypeSyncManager; + return; + } + break; + case lcecConfTypePdoEntry: + if (strcmp(el, "pdoEntry") == 0) { + currConfType = lcecConfTypePdo; + return; + } + break; + } + + fprintf(stderr, "%s: ERROR: unexpected close tag %s found\n", modname, el); + XML_StopParser(parser, 0); +} + +void *getOutputBuffer(size_t len) { + void *p; + + // reallocate if len do not fit into current buffer + while ((outputBufferLen - outputBufferPos) < len) { + size_t new = outputBufferLen + BUFFSIZE; + outputBuffer = realloc(outputBuffer, new); + if (outputBuffer == NULL) { + fprintf(stderr, "%s: ERROR: Couldn't allocate memory for config token\n", modname); + XML_StopParser(parser, 0); + return NULL; + } + outputBufferLen = new; + } + + // initialize memory + p = outputBuffer + outputBufferPos; + memset(p, 0, len); + outputBufferPos += len; + return p; +} + +int parseHexdump(const char *str, uint8_t *buf) { + char c; + int len = 0; + int nib = 0; + uint8_t val = 0; + + while (1) { + c = *(str++); + + // check for seperator or end of string + if (c == 0 || c == ' ') { + if (nib > 0) { + if (buf != NULL) { + *(buf++) = val; + } + len++; + nib = 0; + val = 0; + } + if (c == 0) { + return len; + } + continue; + } + + // get nibble value + if (c >= '0' && c <= '9') { + c = c - '0'; + } else if (c >= 'a' && c <= 'f') { + c = c - 'a' + 10; + } else if (c >= 'A' && c <= 'F') { + c = c - 'A' + 10; + } else { + return -1; + } + + // allow only byte length + if (nib >= 2) { + return -1; + } + + val = (val << 4) + c; + nib++; + } +} + +void parseMasterAttrs(const char **attr) { + LCEC_CONF_MASTER_T *p = getOutputBuffer(sizeof(LCEC_CONF_MASTER_T)); + if (p == NULL) { + return; + } + + p->confType = lcecConfTypeMaster; + while (*attr) { + const char *name = *(attr++); + const char *val = *(attr++); + + // parse index + if (strcmp(name, "idx") == 0) { + p->index = atoi(val); + continue; + } + + // parse name + if (strcmp(name, "name") == 0) { + strncpy(p->name, val, LCEC_CONF_STR_MAXLEN); + p->name[LCEC_CONF_STR_MAXLEN - 1] = 0; + continue; + } + + // parse appTimePeriod + if (strcmp(name, "appTimePeriod") == 0) { + p->appTimePeriod = atol(val); + continue; + } + + // parse refClockSyncCycles + if (strcmp(name, "refClockSyncCycles") == 0) { + p->refClockSyncCycles = atoll(val); + continue; + } + + // handle error + fprintf(stderr, "%s: ERROR: Invalid master attribute %s\n", modname, name); + XML_StopParser(parser, 0); + return; + } + + // set default name + if (p->name[0] == 0) { + snprintf(p->name, LCEC_CONF_STR_MAXLEN, "%d", p->index); + } + + (*(conf_hal_data->master_count))++; + currMaster = p; +} + +void parseSlaveAttrs(const char **attr) { + LCEC_CONF_SLAVE_T *p = getOutputBuffer(sizeof(LCEC_CONF_SLAVE_T)); + if (p == NULL) { + return; + } + + p->confType = lcecConfTypeSlave; + p->type = lcecSlaveTypeInvalid; + while (*attr) { + const char *name = *(attr++); + const char *val = *(attr++); + + // parse index + if (strcmp(name, "idx") == 0) { + p->index = atoi(val); + continue; + } + + // parse slave type + if (strcmp(name, "type") == 0) { + const LCEC_CONF_TYPELIST_T *slaveType; + for (slaveType = slaveTypes; slaveType->name != NULL; slaveType++) { + if (strcmp(val, slaveType->name) == 0) { + break; + } + } + if (slaveType->name == NULL) { + fprintf(stderr, "%s: ERROR: Invalid slave type %s\n", modname, val); + XML_StopParser(parser, 0); + return; + } + p->type = slaveType->type; + continue; + } + + // parse name + if (strcmp(name, "name") == 0) { + strncpy(p->name, val, LCEC_CONF_STR_MAXLEN); + p->name[LCEC_CONF_STR_MAXLEN - 1] = 0; + continue; + } + + // generic only attributes + if (p->type == lcecSlaveTypeGeneric) { + // parse vid (hex value) + if (strcmp(name, "vid") == 0) { + p->vid = strtol(val, NULL, 16); + continue; + } + + // parse pid (hex value) + if (strcmp(name, "pid") == 0) { + p->pid = strtol(val, NULL, 16); + continue; + } + + // parse configPdos + if (strcmp(name, "configPdos") == 0) { + p->configPdos = (strcasecmp(val, "true") == 0); + continue; + } + } + + // handle error + fprintf(stderr, "%s: ERROR: Invalid slave attribute %s\n", modname, name); + XML_StopParser(parser, 0); + return; + } + + // set default name + if (p->name[0] == 0) { + snprintf(p->name, LCEC_CONF_STR_MAXLEN, "%d", p->index); + } + + // type is required + if (p->type == lcecSlaveTypeInvalid) { + fprintf(stderr, "%s: ERROR: Slave has no type attribute\n", modname); + XML_StopParser(parser, 0); + return; + } + + (*(conf_hal_data->slave_count))++; + currSlave = p; +} + +void parseDcConfAttrs(const char **attr) { + LCEC_CONF_DC_T *p = getOutputBuffer(sizeof(LCEC_CONF_DC_T)); + if (p == NULL) { + return; + } + + p->confType = lcecConfTypeDcConf; + while (*attr) { + const char *name = *(attr++); + const char *val = *(attr++); + + // parse assignActivate (hex value) + if (strcmp(name, "assignActivate") == 0) { + p->assignActivate = strtol(val, NULL, 16); + continue; + } + + // parse sync0Cycle + if (strcmp(name, "sync0Cycle") == 0) { + p->sync0Cycle = parseSyncCycle(val); + continue; + } + + // parse sync0Shift + if (strcmp(name, "sync0Shift") == 0) { + p->sync0Shift = atoi(val); + continue; + } + + // parse sync1Cycle + if (strcmp(name, "sync1Cycle") == 0) { + p->sync1Cycle = parseSyncCycle(val); + continue; + } + + // parse sync1Shift + if (strcmp(name, "sync1Shift") == 0) { + p->sync1Shift = atoi(val); + continue; + } + + // handle error + fprintf(stderr, "%s: ERROR: Invalid dcConfig attribute %s\n", modname, name); + XML_StopParser(parser, 0); + return; + } +} + +void parseWatchdogAttrs(const char **attr) { + LCEC_CONF_WATCHDOG_T *p = getOutputBuffer(sizeof(LCEC_CONF_WATCHDOG_T)); + if (p == NULL) { + return; + } + + p->confType = lcecConfTypeWatchdog; + while (*attr) { + const char *name = *(attr++); + const char *val = *(attr++); + + // parse divider + if (strcmp(name, "divider") == 0) { + p->divider = atoi(val); + continue; + } + + // parse intervals + if (strcmp(name, "intervals") == 0) { + p->intervals = atoi(val); + continue; + } + + // handle error + fprintf(stderr, "%s: ERROR: Invalid watchdog attribute %s\n", modname, name); + XML_StopParser(parser, 0); + return; + } +} + +void parseSdoConfigAttrs(const char **attr) { + int tmp; + LCEC_CONF_SDOCONF_T *p = getOutputBuffer(sizeof(LCEC_CONF_SDOCONF_T)); + if (p == NULL) { + return; + } + + p->confType = lcecConfTypeSdoConfig; + p->index = 0xffff; + p->subindex = 0xff; + while (*attr) { + const char *name = *(attr++); + const char *val = *(attr++); + + // parse index + if (strcmp(name, "idx") == 0) { + tmp = strtol(val, NULL, 16); + if (tmp < 0 || tmp >= 0xffff) { + fprintf(stderr, "%s: ERROR: Invalid sdoConfig idx %d\n", modname, tmp); + XML_StopParser(parser, 0); + return; + } + p->index = tmp; + continue; + } + + // parse subIdx + if (strcmp(name, "subIdx") == 0) { + if (strcasecmp(val, "complete") == 0) { + p->subindex = LCEC_CONF_SDO_COMPLETE_SUBIDX; + continue; + } + tmp = strtol(val, NULL, 16); + if (tmp < 0 || tmp >= 0xff) { + fprintf(stderr, "%s: ERROR: Invalid sdoConfig subIdx %d\n", modname, tmp); + XML_StopParser(parser, 0); + return; + } + p->subindex = tmp; + continue; + } + + // handle error + fprintf(stderr, "%s: ERROR: Invalid sdoConfig attribute %s\n", modname, name); + XML_StopParser(parser, 0); + return; + } + + // idx is required + if (p->index == 0xffff) { + fprintf(stderr, "%s: ERROR: sdoConfig has no idx attribute\n", modname); + XML_StopParser(parser, 0); + return; + } + + // subIdx is required + if (p->subindex == 0xff) { + fprintf(stderr, "%s: ERROR: sdoConfig has no subIdx attribute\n", modname); + XML_StopParser(parser, 0); + return; + } + + currSdoConf = p; + currSlave->sdoConfigLength += sizeof(LCEC_CONF_SDOCONF_T); +} + +void parseSdoDataRawAttrs(const char **attr) { + int len; + uint8_t *p; + + while (*attr) { + const char *name = *(attr++); + const char *val = *(attr++); + + // parse data + if (strcmp(name, "data") == 0) { + len = parseHexdump(val, NULL); + if (len < 0) { + fprintf(stderr, "%s: ERROR: Invalid sdoDataRaw data\n", modname); + XML_StopParser(parser, 0); + return; + } + if (len > 0) { + p = (uint8_t *) getOutputBuffer(len); + if (p != NULL) { + parseHexdump(val, p); + currSdoConf->length += len; + currSlave->sdoConfigLength += len; + } + } + continue; + } + + // handle error + fprintf(stderr, "%s: ERROR: Invalid pdoEntry attribute %s\n", modname, name); + XML_StopParser(parser, 0); + return; + } +} + +void parseSyncManagerAttrs(const char **attr) { + int tmp; + LCEC_CONF_SYNCMANAGER_T *p = getOutputBuffer(sizeof(LCEC_CONF_SYNCMANAGER_T)); + if (p == NULL) { + return; + } + + p->confType = lcecConfTypeSyncManager; + p->index = 0xff; + p->dir = EC_DIR_INVALID; + while (*attr) { + const char *name = *(attr++); + const char *val = *(attr++); + + // parse index + if (strcmp(name, "idx") == 0) { + tmp = atoi(val); + if (tmp < 0 || tmp >= EC_MAX_SYNC_MANAGERS) { + fprintf(stderr, "%s: ERROR: Invalid syncManager idx %d\n", modname, tmp); + XML_StopParser(parser, 0); + return; + } + p->index = tmp; + continue; + } + + // parse dir + if (strcmp(name, "dir") == 0) { + if (strcasecmp(val, "in") == 0) { + p->dir = EC_DIR_INPUT; + continue; + } + if (strcasecmp(val, "out") == 0) { + p->dir = EC_DIR_OUTPUT; + continue; + } + fprintf(stderr, "%s: ERROR: Invalid syncManager dir %s\n", modname, val); + XML_StopParser(parser, 0); + return; + } + + // handle error + fprintf(stderr, "%s: ERROR: Invalid syncManager attribute %s\n", modname, name); + XML_StopParser(parser, 0); + return; + } + + // idx is required + if (p->index == 0xff) { + fprintf(stderr, "%s: ERROR: syncManager has no idx attribute\n", modname); + XML_StopParser(parser, 0); + return; + } + + // dir is required + if (p->dir == EC_DIR_INVALID) { + fprintf(stderr, "%s: ERROR: syncManager has no dir attribute\n", modname); + XML_StopParser(parser, 0); + return; + } + + (currSlave->syncManagerCount)++; + currSyncManager = p; +} + +void parsePdoAttrs(const char **attr) { + int tmp; + LCEC_CONF_PDO_T *p = getOutputBuffer(sizeof(LCEC_CONF_PDO_T)); + if (p == NULL) { + return; + } + + p->confType = lcecConfTypePdo; + p->index = 0xffff; + while (*attr) { + const char *name = *(attr++); + const char *val = *(attr++); + + // parse index + if (strcmp(name, "idx") == 0) { + tmp = strtol(val, NULL, 16); + if (tmp < 0 || tmp >= 0xffff) { + fprintf(stderr, "%s: ERROR: Invalid pdo idx %d\n", modname, tmp); + XML_StopParser(parser, 0); + return; + } + p->index = tmp; + continue; + } + + // handle error + fprintf(stderr, "%s: ERROR: Invalid pdo attribute %s\n", modname, name); + XML_StopParser(parser, 0); + return; + } + + // idx is required + if (p->index == 0xffff) { + fprintf(stderr, "%s: ERROR: pdo has no idx attribute\n", modname); + XML_StopParser(parser, 0); + return; + } + + (currSlave->pdoCount)++; + (currSyncManager->pdoCount)++; + currPdo = p; +} + +void parsePdoEntryAttrs(const char **attr) { + int tmp; + int floatReq; + LCEC_CONF_PDOENTRY_T *p = getOutputBuffer(sizeof(LCEC_CONF_PDOENTRY_T)); + if (p == NULL) { + return; + } + + floatReq = 0; + p->confType = lcecConfTypePdoEntry; + p->index = 0xffff; + p->subindex = 0xff; + p->floatScale = 1.0; + while (*attr) { + const char *name = *(attr++); + const char *val = *(attr++); + + // parse index + if (strcmp(name, "idx") == 0) { + tmp = strtol(val, NULL, 16); + if (tmp < 0 || tmp >= 0xffff) { + fprintf(stderr, "%s: ERROR: Invalid pdoEntry idx %d\n", modname, tmp); + XML_StopParser(parser, 0); + return; + } + p->index = tmp; + continue; + } + + // parse subIdx + if (strcmp(name, "subIdx") == 0) { + tmp = strtol(val, NULL, 16); + if (tmp < 0 || tmp >= 0xff) { + fprintf(stderr, "%s: ERROR: Invalid pdoEntry subIdx %d\n", modname, tmp); + XML_StopParser(parser, 0); + return; + } + p->subindex = tmp; + continue; + } + + // parse bitLen + if (strcmp(name, "bitLen") == 0) { + tmp = atoi(val); + if (tmp <= 0 || tmp > 32) { + fprintf(stderr, "%s: ERROR: Invalid pdoEntry bitLen %d\n", modname, tmp); + XML_StopParser(parser, 0); + return; + } + p->bitLength = tmp; + continue; + } + + // parse halType + if (strcmp(name, "halType") == 0) { + if (strcasecmp(val, "bit") == 0) { + p->halType = HAL_BIT; + continue; + } + if (strcasecmp(val, "s32") == 0) { + p->halType = HAL_S32; + continue; + } + if (strcasecmp(val, "u32") == 0) { + p->halType = HAL_U32; + continue; + } + if (strcasecmp(val, "float") == 0) { + p->halType = HAL_FLOAT; + continue; + } + if (strcasecmp(val, "float-unsigned") == 0) { + p->halType = HAL_FLOAT; + p->floatUnsigned = 1; + continue; + } + fprintf(stderr, "%s: ERROR: Invalid pdoEntry halType %s\n", modname, val); + XML_StopParser(parser, 0); + return; + } + + // parse scale + if (strcmp(name, "scale") == 0) { + floatReq = 1; + p->floatScale = atof(val); + } + + // parse offset + if (strcmp(name, "offset") == 0) { + floatReq = 1; + p->floatOffset = atof(val); + } + + // parse halPin + if (strcmp(name, "halPin") == 0) { + strncpy(p->halPin, val, LCEC_CONF_STR_MAXLEN); + p->halPin[LCEC_CONF_STR_MAXLEN - 1] = 0; + continue; + } + + // handle error + fprintf(stderr, "%s: ERROR: Invalid pdoEntry attribute %s\n", modname, name); + XML_StopParser(parser, 0); + return; + } + + // idx is required + if (p->index == 0xffff) { + fprintf(stderr, "%s: ERROR: pdoEntry has no idx attribute\n", modname); + XML_StopParser(parser, 0); + return; + } + + // subIdx is required + if (p->subindex == 0xff) { + fprintf(stderr, "%s: ERROR: pdoEntry has no subIdx attribute\n", modname); + XML_StopParser(parser, 0); + return; + } + + // bitLen is required + if (p->bitLength == 0) { + fprintf(stderr, "%s: ERROR: pdoEntry has no bitLen attribute\n", modname); + XML_StopParser(parser, 0); + return; + } + + // check for float type if required + if (floatReq && p->halType != HAL_FLOAT) { + fprintf(stderr, "%s: ERROR: pdoEntry has scale/offset attributes but pin type is not 'float'\n", modname); + XML_StopParser(parser, 0); + return; + } + + (currSlave->pdoEntryCount)++; + if (p->halPin[0] != 0) { + (currSlave->pdoMappingCount)++; + } + (currPdo->pdoEntryCount)++; +} + +int parseSyncCycle(const char *nptr) { + // chack for master period multiples + if (*nptr == '*') { + nptr++; + return atoi(nptr) * currMaster->appTimePeriod; + } + + // custom value + return atoi(nptr); +} + diff --git a/src/lcec_conf.h b/src/lcec_conf.h new file mode 100644 index 00000000..effc0308 --- /dev/null +++ b/src/lcec_conf.h @@ -0,0 +1,196 @@ +// +// Copyright (C) 2011 Sascha Ittner +// +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2 of the License, or +// (at your option) any later version. +// +// This program 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 this program; if not, write to the Free Software +// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA +// +#ifndef _LCEC_CONF_H_ +#define _LCEC_CONF_H_ + +#include "hal.h" +#include "ecrt.h" + +#define LCEC_MODULE_NAME "lcec" + +#define LCEC_CONF_SHMEM_KEY 0xACB572C7 +#define LCEC_CONF_SHMEM_MAGIC 0x036ED5A3 + +#define LCEC_CONF_STR_MAXLEN 32 + +typedef enum { + lcecConfTypeNone, + lcecConfTypeMasters, + lcecConfTypeMaster, + lcecConfTypeSlave, + lcecConfTypeDcConf, + lcecConfTypeWatchdog, + lcecConfTypeSyncManager, + lcecConfTypePdo, + lcecConfTypePdoEntry, + lcecConfTypeSdoConfig, + lcecConfTypeSdoDataRaw +} LCEC_CONF_TYPE_T; + +typedef enum { + lcecSlaveTypeInvalid, + lcecSlaveTypeGeneric, + lcecSlaveTypeEK1100, + lcecSlaveTypeEL1002, + lcecSlaveTypeEL1004, + lcecSlaveTypeEL1008, + lcecSlaveTypeEL1012, + lcecSlaveTypeEL1014, + lcecSlaveTypeEL1018, + lcecSlaveTypeEL1024, + lcecSlaveTypeEL1034, + lcecSlaveTypeEL1084, + lcecSlaveTypeEL1088, + lcecSlaveTypeEL1094, + lcecSlaveTypeEL1098, + lcecSlaveTypeEL1104, + lcecSlaveTypeEL1114, + lcecSlaveTypeEL1124, + lcecSlaveTypeEL1134, + lcecSlaveTypeEL1144, + lcecSlaveTypeEL1808, + lcecSlaveTypeEL1809, + lcecSlaveTypeEL2002, + lcecSlaveTypeEL2004, + lcecSlaveTypeEL2008, + lcecSlaveTypeEL2022, + lcecSlaveTypeEL2024, + lcecSlaveTypeEL2032, + lcecSlaveTypeEL2034, + lcecSlaveTypeEL2042, + lcecSlaveTypeEL2084, + lcecSlaveTypeEL2088, + lcecSlaveTypeEL2124, + lcecSlaveTypeEL2808, + lcecSlaveTypeEL2809, + lcecSlaveTypeEL3102, + lcecSlaveTypeEL3112, + lcecSlaveTypeEL3122, + lcecSlaveTypeEL3142, + lcecSlaveTypeEL3152, + lcecSlaveTypeEL3162, + lcecSlaveTypeEL4001, + lcecSlaveTypeEL4011, + lcecSlaveTypeEL4021, + lcecSlaveTypeEL4031, + lcecSlaveTypeEL4002, + lcecSlaveTypeEL4012, + lcecSlaveTypeEL4022, + lcecSlaveTypeEL4032, + lcecSlaveTypeEL4102, + lcecSlaveTypeEL4112, + lcecSlaveTypeEL4122, + lcecSlaveTypeEL4132, + lcecSlaveTypeEL5101, + lcecSlaveTypeEL5151, + lcecSlaveTypeEL5152, + lcecSlaveTypeEL2521, + lcecSlaveTypeEL7342, + lcecSlaveTypeEL9505, + lcecSlaveTypeEL9508, + lcecSlaveTypeEL9510, + lcecSlaveTypeEL9512, + lcecSlaveTypeEL9515, + lcecSlaveTypeEM7004, + lcecSlaveTypeStMDS5k, + lcecSlaveTypeDeASDA +} LCEC_SLAVE_TYPE_T; + +typedef struct { + uint32_t magic; + size_t length; +} LCEC_CONF_HEADER_T; + +typedef struct { + LCEC_CONF_TYPE_T confType; + int index; + uint32_t appTimePeriod; + int refClockSyncCycles; + char name[LCEC_CONF_STR_MAXLEN]; +} LCEC_CONF_MASTER_T; + +typedef struct { + LCEC_CONF_TYPE_T confType; + int index; + LCEC_SLAVE_TYPE_T type; + uint32_t vid; + uint32_t pid; + int configPdos; + unsigned int syncManagerCount; + unsigned int pdoCount; + unsigned int pdoEntryCount; + unsigned int pdoMappingCount; + size_t sdoConfigLength; + char name[LCEC_CONF_STR_MAXLEN]; +} LCEC_CONF_SLAVE_T; + +typedef struct { + LCEC_CONF_TYPE_T confType; + uint16_t assignActivate; + uint32_t sync0Cycle; + uint32_t sync0Shift; + uint32_t sync1Cycle; + uint32_t sync1Shift; +} LCEC_CONF_DC_T; + +typedef struct { + LCEC_CONF_TYPE_T confType; + uint16_t divider; + uint16_t intervals; +} LCEC_CONF_WATCHDOG_T; + +typedef struct { + LCEC_CONF_TYPE_T confType; + uint8_t index; + ec_direction_t dir; + unsigned int pdoCount; +} LCEC_CONF_SYNCMANAGER_T; + +typedef struct { + LCEC_CONF_TYPE_T confType; + uint16_t index; + unsigned int pdoEntryCount; +} LCEC_CONF_PDO_T; + +typedef struct { + LCEC_CONF_TYPE_T confType; + uint16_t index; + uint8_t subindex; + uint8_t bitLength; + hal_type_t halType; + int floatUnsigned; + hal_float_t floatScale; + hal_float_t floatOffset; + char halPin[LCEC_CONF_STR_MAXLEN]; +} LCEC_CONF_PDOENTRY_T; + +typedef struct { + LCEC_CONF_TYPE_T confType; +} LCEC_CONF_NULL_T; + +#define LCEC_CONF_SDO_COMPLETE_SUBIDX -1 + +typedef struct { + LCEC_CONF_TYPE_T confType; + uint16_t index; + int16_t subindex; + size_t length; + uint8_t data[]; +} LCEC_CONF_SDOCONF_T; + +#endif diff --git a/src/lcec_deasda.c b/src/lcec_deasda.c new file mode 100644 index 00000000..3a2a61f8 --- /dev/null +++ b/src/lcec_deasda.c @@ -0,0 +1,563 @@ +// +// Copyright (C) 2014 Sascha Ittner +// +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2 of the License, or +// (at your option) any later version. +// +// This program 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 this program; if not, write to the Free Software +// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA +// + +#include "lcec.h" +#include "lcec_deasda.h" + +#define DEASDA_PULSES_PER_REV_DEFLT (1280000) +#define DEASDA_RPM_FACTOR (0.1) +#define DEASDA_RPM_RCPT (1.0 / DEASDA_RPM_FACTOR) +#define DEASDA_RPM_MUL (60.0) +#define DEASDA_RPM_DIV (1.0 / 60.0) + +#define DEASDA_FAULT_AUTORESET_CYCLES 10 + +typedef struct { + int do_init; + + long long pos_cnt; + int32_t last_pos_cnt; + + long long extenc_cnt; + int32_t last_extenc_cnt; + + hal_float_t *vel_fb; + hal_float_t *vel_fb_rpm; + hal_float_t *vel_fb_rpm_abs; + hal_float_t *vel_rpm; + hal_bit_t *ready; + hal_bit_t *switched_on; + hal_bit_t *oper_enabled; + hal_bit_t *fault; + hal_bit_t *volt_enabled; + hal_bit_t *quick_stoped; + hal_bit_t *on_disabled; + hal_bit_t *warning; + hal_bit_t *remote; + hal_bit_t *at_speed; + hal_bit_t *limit_active; + hal_bit_t *zero_speed; + hal_s32_t *enc_raw; + hal_s32_t *extenc_raw; + hal_u32_t *pos_raw_hi; + hal_u32_t *pos_raw_lo; + hal_u32_t *extenc_raw_hi; + hal_u32_t *extenc_raw_lo; + hal_float_t *pos_fb; + hal_float_t *pos_extenc; + hal_bit_t *on_home_neg; + hal_bit_t *on_home_pos; + hal_bit_t *pos_reset; + hal_bit_t *switch_on; + hal_bit_t *enable_volt; + hal_bit_t *quick_stop; + hal_bit_t *enable; + hal_bit_t *fault_reset; + hal_bit_t *halt; + hal_float_t *vel_cmd; + + hal_float_t pos_scale; + hal_float_t extenc_scale; + hal_u32_t pprev; + hal_s32_t home_raw; + hal_u32_t fault_autoreset_cycles; + + hal_float_t pos_scale_old; + hal_u32_t pprev_old; + double pos_scale_rcpt; + double pos_scale_cnt; + + unsigned int status_pdo_os; + unsigned int currpos_pdo_os; + unsigned int currvel_pdo_os; + unsigned int extenc_pdo_os; + unsigned int control_pdo_os; + unsigned int cmdvel_pdo_os; + + hal_bit_t last_switch_on; + hal_bit_t internal_fault; + hal_u32_t fault_reset_pending; + +} lcec_deasda_data_t; + + +static ec_pdo_entry_info_t lcec_deasda_in[] = { + {0x6041, 0x00, 16}, // Status Word + {0x606C, 0x00, 32}, // Current Velocity + {0x6064, 0x00, 32}, // Current Position + {0x2511, 0x00, 32} // external encoder +}; + +static ec_pdo_entry_info_t lcec_deasda_out[] = { + {0x6040, 0x00, 16}, // Control Word + {0x60FF, 0x00, 32} // Target Velocity +}; + +static ec_pdo_info_t lcec_deasda_pdos_out[] = { + {0x1602, 2, lcec_deasda_out} +}; + +static ec_pdo_info_t lcec_deasda_pdos_in[] = { + {0x1a02, 4, lcec_deasda_in} +}; + +static ec_sync_info_t lcec_deasda_syncs[] = { + {0, EC_DIR_OUTPUT, 0, NULL}, + {1, EC_DIR_INPUT, 0, NULL}, + {2, EC_DIR_OUTPUT, 1, lcec_deasda_pdos_out}, + {3, EC_DIR_INPUT, 1, lcec_deasda_pdos_in}, + {0xff} +}; + +void lcec_deasda_check_scales(lcec_deasda_data_t *hal_data); + +void lcec_deasda_read(struct lcec_slave *slave, long period); +void lcec_deasda_write(struct lcec_slave *slave, long period); + +int lcec_deasda_init(int comp_id, struct lcec_slave *slave, ec_pdo_entry_reg_t *pdo_entry_regs) { + lcec_master_t *master = slave->master; + lcec_deasda_data_t *hal_data; + int err; + uint32_t tu; + int8_t ti; + + // initialize callbacks + slave->proc_read = lcec_deasda_read; + slave->proc_write = lcec_deasda_write; + + // alloc hal memory + if ((hal_data = hal_malloc(sizeof(lcec_deasda_data_t))) == NULL) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "hal_malloc() for slave %s.%s failed\n", master->name, slave->name); + return -EIO; + } + memset(hal_data, 0, sizeof(lcec_deasda_data_t)); + slave->hal_data = hal_data; + + // set to cyclic synchronous velocity mode + if (ecrt_slave_config_sdo8(slave->config, 0x6060, 0x00, 9) != 0) { + rtapi_print_msg (RTAPI_MSG_ERR, LCEC_MSG_PFX "fail to configure slave %s.%s sdo velo mode\n", master->name, slave->name); + } + + // set interpolation time period + tu = master->app_time_period; + ti = -9; + while ((tu % 10) == 0 || tu > 255) { tu /= 10; ti++; } + if (ecrt_slave_config_sdo8(slave->config, 0x60C2, 0x01, (uint8_t)tu) != 0) { + rtapi_print_msg (RTAPI_MSG_ERR, LCEC_MSG_PFX "fail to configure slave %s.%s sdo ipol time period units\n", master->name, slave->name); + } + if (ecrt_slave_config_sdo8(slave->config, 0x60C2, 0x02, ti) != 0) { + rtapi_print_msg (RTAPI_MSG_ERR, LCEC_MSG_PFX "fail to configure slave %s.%s sdo ipol time period index\n", master->name, slave->name); + } + + // initialize sync info + slave->sync_info = lcec_deasda_syncs; + + // initialize POD entries + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6041, 0x00, &hal_data->status_pdo_os, NULL); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x606C, 0x00, &hal_data->currvel_pdo_os, NULL); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6064, 0x00, &hal_data->currpos_pdo_os, NULL); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x2511, 0x00, &hal_data->extenc_pdo_os, NULL); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6040, 0x00, &hal_data->control_pdo_os, NULL); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x60FF, 0x00, &hal_data->cmdvel_pdo_os, NULL); + + // export pins + if ((err = hal_pin_float_newf(HAL_OUT, &(hal_data->vel_fb), comp_id, "%s.%s.%s.srv-vel-fb", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-vel-fb failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_float_newf(HAL_OUT, &(hal_data->vel_fb_rpm), comp_id, "%s.%s.%s.srv-vel-fb-rpm", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-vel-fb-rpm failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_float_newf(HAL_OUT, &(hal_data->vel_fb_rpm_abs), comp_id, "%s.%s.%s.srv-vel-fb-rpm-abs", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-vel-fb-rpm-abs failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_float_newf(HAL_OUT, &(hal_data->vel_rpm), comp_id, "%s.%s.%s.srv-vel-rpm", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-vel-rpm failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(hal_data->ready), comp_id, "%s.%s.%s.srv-ready", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-ready failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(hal_data->switched_on), comp_id, "%s.%s.%s.srv-switched-on", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-switched-on failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(hal_data->oper_enabled), comp_id, "%s.%s.%s.srv-oper-enabled", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-oper-enabled failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(hal_data->fault), comp_id, "%s.%s.%s.srv-fault", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-fault failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(hal_data->volt_enabled), comp_id, "%s.%s.%s.srv-volt-enabled", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-volt-enabled failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(hal_data->quick_stoped), comp_id, "%s.%s.%s.srv-quick-stoped", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-quick-stoped failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(hal_data->on_disabled), comp_id, "%s.%s.%s.srv-on-disabled", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-on-disabled failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(hal_data->warning), comp_id, "%s.%s.%s.srv-warning", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-warning failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(hal_data->remote), comp_id, "%s.%s.%s.srv-remote", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-remote failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(hal_data->at_speed), comp_id, "%s.%s.%s.srv-at-speed", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-at-speed failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(hal_data->limit_active), comp_id, "%s.%s.%s.srv-limit-active", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-limit-active failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(hal_data->zero_speed), comp_id, "%s.%s.%s.srv-zero-speed", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-zero-speed failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_s32_newf(HAL_OUT, &(hal_data->enc_raw), comp_id, "%s.%s.%s.srv-enc-raw", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-enc-raw failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_u32_newf(HAL_OUT, &(hal_data->pos_raw_hi), comp_id, "%s.%s.%s.srv-pos-raw-hi", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-pos-raw-hi failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_u32_newf(HAL_OUT, &(hal_data->pos_raw_lo), comp_id, "%s.%s.%s.srv-pos-raw-lo", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-pos-raw-lo failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_s32_newf(HAL_OUT, &(hal_data->extenc_raw), comp_id, "%s.%s.%s.srv-extenc-raw", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-extenc-raw failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_u32_newf(HAL_OUT, &(hal_data->extenc_raw_hi), comp_id, "%s.%s.%s.srv-extenc-raw-hi", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-extenc-raw-hi failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_u32_newf(HAL_OUT, &(hal_data->extenc_raw_lo), comp_id, "%s.%s.%s.srv-extenc-raw-lo", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-extenc-raw-lo failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_float_newf(HAL_OUT, &(hal_data->pos_fb), comp_id, "%s.%s.%s.srv-pos-fb", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-pos-fb failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_float_newf(HAL_OUT, &(hal_data->pos_extenc), comp_id, "%s.%s.%s.srv-pos-extenc", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-pos-extenc failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(hal_data->on_home_neg), comp_id, "%s.%s.%s.srv-on-home-neg", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-on-home-neg failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(hal_data->on_home_pos), comp_id, "%s.%s.%s.srv-on-home-pos", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-on-home-pos failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_IN, &(hal_data->pos_reset), comp_id, "%s.%s.%s.srv-pos-reset", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-pos-reset failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_IN, &(hal_data->switch_on), comp_id, "%s.%s.%s.srv-switch-on", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-switch-on failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_IN, &(hal_data->enable_volt), comp_id, "%s.%s.%s.srv-enable-volt", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-enable-volt failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_IN, &(hal_data->quick_stop), comp_id, "%s.%s.%s.srv-quick-stop", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-quick-stop failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_IN, &(hal_data->enable), comp_id, "%s.%s.%s.srv-enable", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-enable failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_IN, &(hal_data->fault_reset), comp_id, "%s.%s.%s.srv-fault-reset", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-fault-reset failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_IN, &(hal_data->halt), comp_id, "%s.%s.%s.srv-halt", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-halt failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_float_newf(HAL_IN, &(hal_data->vel_cmd), comp_id, "%s.%s.%s.srv-vel-cmd", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-vel-cmd failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + + // export parameters + if ((err = hal_param_float_newf(HAL_RW, &(hal_data->pos_scale), comp_id, "%s.%s.%s.srv-pos-scale", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting param %s.%s.%s.srv-pos-scale failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_param_float_newf(HAL_RW, &(hal_data->extenc_scale), comp_id, "%s.%s.%s.srv-extenc-scale", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting param %s.%s.%s.srv-extenc-scale failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_param_u32_newf(HAL_RW, &(hal_data->pprev), comp_id, "%s.%s.%s.srv-pulses-per-rev", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting param %s.%s.%s.srv-pulses-per-rev failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_param_s32_newf(HAL_RW, &(hal_data->home_raw), comp_id, "%s.%s.%s.srv-home-raw", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting param %s.%s.%s.srv-home-raw failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_param_u32_newf(HAL_RW, &(hal_data->fault_autoreset_cycles), comp_id, "%s.%s.%s.srv-fault-autoreset-cycles", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting param %s.%s.%s.srv-fault-autoreset-cycles failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + + // set default pin values + *(hal_data->vel_fb) = 0.0; + *(hal_data->vel_fb_rpm) = 0.0; + *(hal_data->vel_fb_rpm_abs) = 0.0; + *(hal_data->vel_rpm) = 0.0; + *(hal_data->ready) = 0; + *(hal_data->switched_on) = 0; + *(hal_data->oper_enabled) = 0; + *(hal_data->fault) = 0; + *(hal_data->volt_enabled) = 0; + *(hal_data->quick_stoped) = 0; + *(hal_data->on_disabled) = 0; + *(hal_data->warning) = 0; + *(hal_data->remote) = 0; + *(hal_data->at_speed) = 0; + *(hal_data->limit_active) = 0; + *(hal_data->zero_speed) = 0; + *(hal_data->enc_raw) = 0; + *(hal_data->pos_raw_hi) = 0; + *(hal_data->pos_raw_lo) = 0; + *(hal_data->extenc_raw) = 0; + *(hal_data->extenc_raw_hi) = 0; + *(hal_data->extenc_raw_lo) = 0; + *(hal_data->pos_fb) = 0.0; + *(hal_data->pos_extenc) = 0.0; + *(hal_data->on_home_neg) = 0; + *(hal_data->on_home_pos) = 0; + *(hal_data->pos_reset) = 0; + *(hal_data->switch_on) = 0; + *(hal_data->enable_volt) = 0; + *(hal_data->quick_stop) = 0; + *(hal_data->enable) = 0; + *(hal_data->fault_reset) = 0; + *(hal_data->halt) = 0; + *(hal_data->vel_cmd) = 0.0; + + // initialize variables + hal_data->do_init = 1; + hal_data->pos_scale = 1.0; + hal_data->extenc_scale = 1.0; + hal_data->home_raw = 0; + hal_data->fault_autoreset_cycles = DEASDA_FAULT_AUTORESET_CYCLES; + hal_data->pos_cnt = 0; + hal_data->last_pos_cnt = 0; + hal_data->extenc_cnt = 0; + hal_data->last_extenc_cnt = 0; + hal_data->pos_scale_old = hal_data->pos_scale + 1.0; + hal_data->pos_scale_rcpt = 1.0; + hal_data->pos_scale_cnt = 1.0; + hal_data->pprev = DEASDA_PULSES_PER_REV_DEFLT; + hal_data->pprev_old = hal_data->pprev + 1; + hal_data->last_switch_on = 0; + hal_data->internal_fault = 0; + hal_data->fault_reset_pending = 0; + + return 0; +} + +void lcec_deasda_check_scales(lcec_deasda_data_t *hal_data) { + // check for change in scale value + if (hal_data->pos_scale != hal_data->pos_scale_old || hal_data->pprev != hal_data->pprev_old) { + + // scale value has changed, test and update it + if ((hal_data->pos_scale < 1e-20) && (hal_data->pos_scale > -1e-20)) { + // value too small, divide by zero is a bad thing + hal_data->pos_scale = 1.0; + } + + // save new scale to detect future changes + hal_data->pos_scale_old = hal_data->pos_scale; + hal_data->pprev_old = hal_data->pprev; + + // we actually want the reciprocal + hal_data->pos_scale_rcpt = 1.0 / hal_data->pos_scale; + + // scale for counter + hal_data->pos_scale_cnt = hal_data->pos_scale / ((double) hal_data->pprev); + } +} + +void lcec_deasda_read(struct lcec_slave *slave, long period) { + lcec_master_t *master = slave->master; + lcec_deasda_data_t *hal_data = (lcec_deasda_data_t *) slave->hal_data; + uint8_t *pd = master->process_data; + uint16_t status; + int32_t speed_raw; + double rpm; + int32_t pos_cnt, pos_cnt_diff; + + // wait for slave to be operational + if (!slave->state.operational) { + return; + } + + // check for change in scale value + lcec_deasda_check_scales(hal_data); + + // read status word + status = EC_READ_U16(&pd[hal_data->status_pdo_os]); + *(hal_data->ready) = (status >> 0) & 0x01; + *(hal_data->switched_on) = (status >> 1) & 0x01; + *(hal_data->oper_enabled) = (status >> 2) & 0x01; + hal_data->internal_fault = (status >> 3) & 0x01; + *(hal_data->volt_enabled) = (status >> 4) & 0x01; + *(hal_data->quick_stoped) = !((status >> 5) & 0x01); + *(hal_data->on_disabled) = (status >> 6) & 0x01; + *(hal_data->warning) = (status >> 7) & 0x01; + *(hal_data->remote) = (status >> 9) & 0x01; + *(hal_data->at_speed) = (status >> 10) & 0x01; + *(hal_data->limit_active) = (status >> 11) & 0x01; + *(hal_data->zero_speed) = (status >> 12) & 0x01; + + // clear pending fault reset if no fault + if (!hal_data->internal_fault) { + hal_data->fault_reset_pending = 0; + } + + // generate gated fault + if (hal_data->fault_reset_pending > 0) { + hal_data->fault_reset_pending--; + *(hal_data->fault) = 0; + } else { + *(hal_data->fault) = hal_data->internal_fault; + } + + // read current speed + speed_raw = EC_READ_S32(&pd[hal_data->currvel_pdo_os]); + rpm = (double)speed_raw * DEASDA_RPM_FACTOR; + *(hal_data->vel_fb_rpm) = rpm; + *(hal_data->vel_fb_rpm_abs) = fabs(rpm); + *(hal_data->vel_fb) = rpm * DEASDA_RPM_DIV * hal_data->pos_scale; + + // update raw position counter + pos_cnt = EC_READ_S32(&pd[hal_data->currpos_pdo_os]); + *(hal_data->enc_raw) = pos_cnt; + *(hal_data->on_home_neg) = (pos_cnt <= hal_data->home_raw); + *(hal_data->on_home_pos) = (pos_cnt >= hal_data->home_raw); + pos_cnt_diff = pos_cnt - hal_data->last_pos_cnt; + hal_data->last_pos_cnt = pos_cnt; + hal_data->pos_cnt += pos_cnt_diff; + + // update external encoder counter + pos_cnt = EC_READ_S32(&pd[hal_data->extenc_pdo_os]); + *(hal_data->extenc_raw) = pos_cnt; + pos_cnt_diff = pos_cnt - hal_data->last_extenc_cnt; + hal_data->last_extenc_cnt = pos_cnt; + hal_data->extenc_cnt += pos_cnt_diff; + + // handle initialization + if (hal_data->do_init || *(hal_data->pos_reset)) { + hal_data->do_init = 0; + hal_data->pos_cnt = 0; + hal_data->extenc_cnt = 0; + } + + // update raw counter pins + *(hal_data->pos_raw_hi) = (hal_data->pos_cnt >> 32) & 0xffffffff; + *(hal_data->pos_raw_lo) = (hal_data->pos_cnt) & 0xffffffff; + *(hal_data->extenc_raw_hi) = (hal_data->extenc_cnt >> 32) & 0xffffffff; + *(hal_data->extenc_raw_lo) = (hal_data->extenc_cnt) & 0xffffffff; + + // scale count to make floating point position + *(hal_data->pos_fb) = (double)(hal_data->pos_cnt) * hal_data->pos_scale_cnt; + *(hal_data->pos_extenc) = (double)(hal_data->extenc_cnt) * hal_data->extenc_scale; +} + +void lcec_deasda_write(struct lcec_slave *slave, long period) { + lcec_master_t *master = slave->master; + lcec_deasda_data_t *hal_data = (lcec_deasda_data_t *) slave->hal_data; + uint8_t *pd = master->process_data; + uint16_t control; + double speed_raw; + int switch_on_edge; + + // check for enable edge + switch_on_edge = *(hal_data->switch_on) && !hal_data->last_switch_on; + hal_data->last_switch_on = *(hal_data->switch_on); + + // check for autoreset + if (hal_data->fault_autoreset_cycles > 0 && switch_on_edge && hal_data->internal_fault) { + hal_data->fault_reset_pending = hal_data->fault_autoreset_cycles; + } + + // check for change in scale value + lcec_deasda_check_scales(hal_data); + + // write dev ctrl + control = 0; + if (*(hal_data->enable_volt)) { + control |= (1 << 1); + } + if (! *(hal_data->quick_stop)) { + control |= (1 << 2); + } + if (*(hal_data->fault_reset)) { + control |= (1 << 7); + } + if (*(hal_data->halt)) { + control |= (1 << 8); + } + if (hal_data->fault_reset_pending > 0) { + control |= (1 << 7); + } else { + if (*(hal_data->switch_on)) { + control |= (1 << 0); + } + if (*(hal_data->enable) && *(hal_data->switched_on)) { + control |= (1 << 3); + } + } + EC_WRITE_U16(&pd[hal_data->control_pdo_os], control); + + // calculate rpm command + *(hal_data->vel_rpm) = *(hal_data->vel_cmd) * hal_data->pos_scale_rcpt * DEASDA_RPM_MUL; + + // set RPM + speed_raw = *(hal_data->vel_rpm) * DEASDA_RPM_RCPT; + if (speed_raw > (double)0x7fffffff) { + speed_raw = (double)0x7fffffff; + } + if (speed_raw < (double)-0x7fffffff) { + speed_raw = (double)-0x7fffffff; + } + EC_WRITE_S32(&pd[hal_data->cmdvel_pdo_os], (int32_t)speed_raw); +} + diff --git a/src/lcec_deasda.h b/src/lcec_deasda.h new file mode 100644 index 00000000..27bbf38e --- /dev/null +++ b/src/lcec_deasda.h @@ -0,0 +1,148 @@ +// +// Copyright (C) 2014 Sascha Ittner +// +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2 of the License, or +// (at your option) any later version. +// +// This program 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 this program; if not, write to the Free Software +// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA +// + +/*############################################################################# + +Description of Pins/Parameters: + +Pins: + FLOAT OUT srv-vel-fb + Actual velocity feddback in scale units per second. + + FLOAT OUT srv-vel-fb-rpm + Actual velocity feddback in revolutions per minute. + + FLOAT OUT srv-vel-fb-rpm-abs + Actual absolute value of velocity feddback in + revolutions per minute. + + FLOAT OUT srv-vel-rpm + Current velocity command send to motor driver in + revolutions per minute. + + BIT OUT srv-ready + ready status flag from motor driver + + BIT OUT srv-switched-on + switched-on status flag from motor driver + + BIT OUT srv-oper-enabled + operation-enabled status flag from motor driver + + BIT OUT srv-fault + fault status flag from motor driver + + BIT OUT srv-volt-enabled + voltage-enabled status flag from motor driver + + BIT OUT srv-quick-stoped + quick-stoped status flag from motor driver + + BIT OUT srv-on-disabled + on-disabled status flag from motor driver + + BIT OUT srv-warning + warning status flag from motor driver + + BIT OUT srv-remote + motor driver is remote controlled + + BIT OUT srv-at-speed + motor has reached requested velocity + + BIT OUT srv-limit-active + motor driver has reached some limit. + + BIT OUT srv-zero-speed + motor has stopped + + S32 OUT srv-enc-raw + raw value from motor encoder without offset calculation. This + value is the unmodified one ad repoted by the motor driver. + + U32 OUT srv-pos-raw-hi + Upper 32 bits of range extended 64 bit encoder value. This is a relative + value wich is reset on restart/reset pin trigger. + + U32 OUT srv-pos-raw-lo + Lower 32 bits of range extended 64 bit encoder value. This is a relative + value wich is reset on restart/reset pin trigger. + + FLOAT OUT srv-pos-fb + Actual encoder position + + BIT OUT srv-on-home-neg + Motor encoder is below virtual home switch position. (See parameter + srv-home-raw) + + BIT OUT srv-on-home-pos + Motor encoder is above virtual home switch position. (See parameter + srv-home-raw) + + BIT IN srv-pos-reset + Reset relative position counter (srv-pos-raw-hi/srv-pos-raw-lo) + + BIT IN srv-switch-on + Switch motor driver on + + BIT IN srv-enable-volt + Enable motor driver voltage + + BIT IN srv-quick-stop + Enable motor quick stop + + BIT IN srv-enable + Enable motor driver operation. This signal is internaly delayed + until driver reports switched-on. + + BIT IN srv-fault-reset + Reset motor driver error + + BIT IN srv-fault-autoreset + If set to 1 errors get automatically reset if drive is disabled (enable = 0) + + BIT IN srv-halt + Set motor driver halt flag + + FLOAT IN srv-vel-cmd + Velocity command input in scale units per second. + +Parameters: + RW FLOAT srv-pos-scale + Scale for position/velocity values (1.0 -> 1 revolution per second) + + RW S32 srv-home-raw + Absolute position of virtual home switch. This could be used to implement + a home switch on motors with abolute encoders without a hardware switch. + +#############################################################################*/ + +#ifndef _LCEC_DEASDA_H_ +#define _LCEC_DEASDA_H_ + +#include "lcec.h" + +#define LCEC_DEASDA_VID LCEC_DELTA_VID +#define LCEC_DEASDA_PID 0x10305070 + +#define LCEC_DEASDA_PDOS 6 + +int lcec_deasda_init(int comp_id, struct lcec_slave *slave, ec_pdo_entry_reg_t *pdo_entry_regs); + +#endif + diff --git a/src/lcec_ek1100.h b/src/lcec_ek1100.h new file mode 100644 index 00000000..e83bd225 --- /dev/null +++ b/src/lcec_ek1100.h @@ -0,0 +1,26 @@ +// +// Copyright (C) 2011 Sascha Ittner +// +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2 of the License, or +// (at your option) any later version. +// +// This program 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 this program; if not, write to the Free Software +// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA +// +#ifndef _LCEC_EK1100_H_ +#define _LCEC_EK1100_H_ + +#define LCEC_EK1100_VID LCEC_BECKHOFF_VID +#define LCEC_EK1100_PID 0x044C2C52 +#define LCEC_EK1100_PDOS 0 + +#endif + diff --git a/src/lcec_el1xxx.c b/src/lcec_el1xxx.c new file mode 100644 index 00000000..e01bcb34 --- /dev/null +++ b/src/lcec_el1xxx.c @@ -0,0 +1,91 @@ +// +// Copyright (C) 2011 Sascha Ittner +// +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2 of the License, or +// (at your option) any later version. +// +// This program 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 this program; if not, write to the Free Software +// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA +// + +#include "lcec.h" +#include "lcec_el1xxx.h" + +typedef struct { + hal_bit_t *in; + hal_bit_t *in_not; + unsigned int pdo_os; + unsigned int pdo_bp; +} lcec_el1xxx_pin_t; + +void lcec_el1xxx_read(struct lcec_slave *slave, long period); + +int lcec_el1xxx_init(int comp_id, struct lcec_slave *slave, ec_pdo_entry_reg_t *pdo_entry_regs) { + lcec_master_t *master = slave->master; + lcec_el1xxx_pin_t *hal_data; + lcec_el1xxx_pin_t *pin; + int i; + int err; + + // initialize callbacks + slave->proc_read = lcec_el1xxx_read; + + // alloc hal memory + if ((hal_data = hal_malloc(sizeof(lcec_el1xxx_pin_t) * slave->pdo_entry_count)) == NULL) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "hal_malloc() for slave %s.%s failed\n", master->name, slave->name); + return -EIO; + } + memset(hal_data, 0, sizeof(lcec_el1xxx_pin_t) * slave->pdo_entry_count); + slave->hal_data = hal_data; + + // initialize pins + for (i=0, pin=hal_data; ipdo_entry_count; i++, pin++) { + // initialize POD entry + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6000 + (i << 4), 0x01, &pin->pdo_os, &pin->pdo_bp); + + // export pins + if ((err = hal_pin_bit_newf(HAL_OUT, &(pin->in), comp_id, "%s.%s.%s.din-%d", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.din-%02d failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(pin->in_not), comp_id, "%s.%s.%s.din-%d-not", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.din-%02d-not failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + + // initialize pins + *(pin->in) = 0; + *(pin->in_not) = 0; + } + + return 0; +} + +void lcec_el1xxx_read(struct lcec_slave *slave, long period) { + lcec_master_t *master = slave->master; + lcec_el1xxx_pin_t *hal_data = (lcec_el1xxx_pin_t *) slave->hal_data; + uint8_t *pd = master->process_data; + lcec_el1xxx_pin_t *pin; + int i, s; + + // wait for slave to be operational + if (!slave->state.operational) { + return; + } + + // check inputs + for (i=0, pin=hal_data; ipdo_entry_count; i++, pin++) { + s = EC_READ_BIT(&pd[pin->pdo_os], pin->pdo_bp); + *(pin->in) = s; + *(pin->in_not) = !s; + } +} + diff --git a/src/lcec_el1xxx.h b/src/lcec_el1xxx.h new file mode 100644 index 00000000..5e76e382 --- /dev/null +++ b/src/lcec_el1xxx.h @@ -0,0 +1,68 @@ +// +// Copyright (C) 2011 Sascha Ittner +// +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2 of the License, or +// (at your option) any later version. +// +// This program 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 this program; if not, write to the Free Software +// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA +// +#ifndef _LCEC_EL1XXX_H_ +#define _LCEC_EL1XXX_H_ + +#include "lcec.h" + +#define LCEC_EL1xxx_VID LCEC_BECKHOFF_VID + +#define LCEC_EL1002_PID 0x03EA3052 +#define LCEC_EL1004_PID 0x03EC3052 +#define LCEC_EL1008_PID 0x03F03052 +#define LCEC_EL1012_PID 0x03F43052 +#define LCEC_EL1014_PID 0x03F63052 +#define LCEC_EL1018_PID 0x03FA3052 +#define LCEC_EL1024_PID 0x04003052 +#define LCEC_EL1034_PID 0x040A3052 +#define LCEC_EL1084_PID 0x043C3052 +#define LCEC_EL1088_PID 0x04403052 +#define LCEC_EL1094_PID 0x04463052 +#define LCEC_EL1098_PID 0x044A3052 +#define LCEC_EL1104_PID 0x04503052 +#define LCEC_EL1114_PID 0x045A3052 +#define LCEC_EL1124_PID 0x04643052 +#define LCEC_EL1134_PID 0x046E3052 +#define LCEC_EL1144_PID 0x04783052 +#define LCEC_EL1808_PID 0x07103052 +#define LCEC_EL1809_PID 0x07113052 + +#define LCEC_EL1002_PDOS 2 +#define LCEC_EL1004_PDOS 4 +#define LCEC_EL1008_PDOS 8 +#define LCEC_EL1012_PDOS 2 +#define LCEC_EL1014_PDOS 4 +#define LCEC_EL1018_PDOS 8 +#define LCEC_EL1024_PDOS 4 +#define LCEC_EL1034_PDOS 4 +#define LCEC_EL1084_PDOS 4 +#define LCEC_EL1088_PDOS 8 +#define LCEC_EL1094_PDOS 4 +#define LCEC_EL1098_PDOS 8 +#define LCEC_EL1104_PDOS 4 +#define LCEC_EL1114_PDOS 4 +#define LCEC_EL1124_PDOS 4 +#define LCEC_EL1134_PDOS 4 +#define LCEC_EL1144_PDOS 4 +#define LCEC_EL1808_PDOS 8 +#define LCEC_EL1809_PDOS 16 + +int lcec_el1xxx_init(int comp_id, struct lcec_slave *slave, ec_pdo_entry_reg_t *pdo_entry_regs); + +#endif + diff --git a/src/lcec_el2521.c b/src/lcec_el2521.c new file mode 100644 index 00000000..33520135 --- /dev/null +++ b/src/lcec_el2521.c @@ -0,0 +1,369 @@ +// +// Copyright (C) 2011 Sascha Ittner +// +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2 of the License, or +// (at your option) any later version. +// +// This program 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 this program; if not, write to the Free Software +// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA +// + +#include "hal.h" + +#include "lcec.h" +#include "lcec_el2521.h" + +typedef struct { + hal_s32_t *count; // pin: captured feedback in counts + hal_float_t *pos_fb; // pin: position feedback (position units) + hal_bit_t *ramp_active; // pin: ramp currently active + hal_bit_t *ramp_disable; // pin: disable ramp + hal_bit_t *in_z; // pin: input z + hal_bit_t *in_z_not; // pin: input z inverted + hal_bit_t *in_t; // pin: input t + hal_bit_t *in_t_not; // pin: input t inverted + + hal_bit_t *enable; // pin for enable stepgen + hal_float_t *vel_cmd; // pin: velocity command (pos units/sec) + + hal_float_t pos_scale; // param: steps per position unit + hal_float_t freq; // param: current frequency + hal_float_t maxvel; // param: max velocity, (pos units/sec) + hal_float_t maxaccel_rise; // param: max accel (pos units/sec^2) + hal_float_t maxaccel_fall; // param: max accel (pos units/sec^2) + + int last_operational; + int16_t last_hw_count; // last hw counter value + double old_scale; // stored scale value + double scale_recip; // reciprocal value used for scaling + + unsigned int state_pdo_os; + unsigned int count_pdo_os; + unsigned int ctrl_pdo_os; + unsigned int freq_pdo_os; + + ec_sdo_request_t *sdo_req_base_freq; + ec_sdo_request_t *sdo_req_max_freq; + ec_sdo_request_t *sdo_req_ramp_rise; + ec_sdo_request_t *sdo_req_ramp_fall; + ec_sdo_request_t *sdo_req_ramp_factor; + + uint32_t sdo_base_freq; + uint16_t sdo_max_freq; + uint16_t sdo_ramp_rise; + uint16_t sdo_ramp_fall; + uint16_t sdo_ramp_factor; + + double freqscale; + double freqscale_recip; + double max_freq; + double max_ac_rise; + double max_ac_fall; + +} lcec_el2521_data_t; + +static ec_pdo_entry_info_t lcec_el2521_in[] = { + {0x6000, 0x01, 16}, // state word + {0x6000, 0x02, 16} // counter value +}; + +static ec_pdo_entry_info_t lcec_el2521_out[] = { + {0x7000, 0x01, 16}, // control word + {0x7000, 0x02, 16} // frequency value +}; + +static ec_pdo_info_t lcec_el2521_pdos_in[] = { + {0x1A00, 2, lcec_el2521_in}, +}; + +static ec_pdo_info_t lcec_el2521_pdos_out[] = { + {0x1600, 2, lcec_el2521_out}, +}; + +static ec_sync_info_t lcec_el2521_syncs[] = { + {0, EC_DIR_OUTPUT, 0, NULL}, + {1, EC_DIR_INPUT, 0, NULL}, + {2, EC_DIR_OUTPUT, 1, lcec_el2521_pdos_out}, + {3, EC_DIR_INPUT, 1, lcec_el2521_pdos_in}, + {0xff} +}; + + +void lcec_el2521_check_scale(lcec_el2521_data_t *hal_data); + +void lcec_el2521_read(struct lcec_slave *slave, long period); +void lcec_el2521_write(struct lcec_slave *slave, long period); + +int lcec_el2521_init(int comp_id, struct lcec_slave *slave, ec_pdo_entry_reg_t *pdo_entry_regs) { + lcec_master_t *master = slave->master; + lcec_el2521_data_t *hal_data; + int err; + double ramp_factor; + + // initialize callbacks + slave->proc_read = lcec_el2521_read; + slave->proc_write = lcec_el2521_write; + + // alloc hal memory + if ((hal_data = hal_malloc(sizeof(lcec_el2521_data_t))) == NULL) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "hal_malloc() for slave %s.%s failed\n", master->name, slave->name); + return -EIO; + } + memset(hal_data, 0, sizeof(lcec_el2521_data_t)); + slave->hal_data = hal_data; + + // read sdos + if ((hal_data->sdo_req_base_freq = lcec_read_sdo(slave, 0x8001, 0x02, 4)) == NULL) { + return -EIO; + } + hal_data->sdo_base_freq = EC_READ_U32(ecrt_sdo_request_data(hal_data->sdo_req_base_freq)); + if ((hal_data->sdo_req_ramp_rise = lcec_read_sdo(slave, 0x8001, 0x04, 2)) == NULL) { + return -EIO; + } + hal_data->sdo_ramp_rise = EC_READ_U16(ecrt_sdo_request_data(hal_data->sdo_req_ramp_rise)); + if ((hal_data->sdo_req_ramp_fall = lcec_read_sdo(slave, 0x8001, 0x05, 2)) == NULL) { + return -EIO; + } + hal_data->sdo_ramp_fall = EC_READ_U16(ecrt_sdo_request_data(hal_data->sdo_req_ramp_fall)); + if ((hal_data->sdo_req_ramp_factor = lcec_read_sdo(slave, 0x8000, 0x07, 2)) == NULL) { + return -EIO; + } + hal_data->sdo_ramp_factor = EC_READ_U16(ecrt_sdo_request_data(hal_data->sdo_req_ramp_factor)); + if ((hal_data->sdo_req_max_freq = lcec_read_sdo(slave, 0x8800, 0x02, 2)) == NULL) { + return -EIO; + } + hal_data->sdo_max_freq = EC_READ_U16(ecrt_sdo_request_data(hal_data->sdo_req_max_freq)); + + // initializer sync info + slave->sync_info = lcec_el2521_syncs; + + // initialize POD entries + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6000, 0x01, &hal_data->state_pdo_os, NULL); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6000, 0x02, &hal_data->count_pdo_os, NULL); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x7000, 0x01, &hal_data->ctrl_pdo_os, NULL); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x7000, 0x02, &hal_data->freq_pdo_os, NULL); + + // export pins + if ((err = hal_pin_s32_newf(HAL_OUT, &(hal_data->count), comp_id, "%s.%s.%s.stp-counts", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.stp-counts failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_float_newf(HAL_OUT, &(hal_data->pos_fb), comp_id, "%s.%s.%s.stp-pos-fb", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.stp-pos-fb failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(hal_data->ramp_active), comp_id, "%s.%s.%s.stp-ramp-active", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.stp-ramp-active failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_IN, &(hal_data->ramp_disable), comp_id, "%s.%s.%s.stp-ramp-disable", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.stp-ramp-disable failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(hal_data->in_z), comp_id, "%s.%s.%s.stp-in-z", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.stp-in-z failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(hal_data->in_z_not), comp_id, "%s.%s.%s.stp-in-z-not", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.stp-in-z-not failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(hal_data->in_t), comp_id, "%s.%s.%s.stp-in-t", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.stp-in-t failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(hal_data->in_t_not), comp_id, "%s.%s.%s.stp-in-t-not", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.stp-in-t-not failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_IN, &(hal_data->enable), comp_id, "%s.%s.%s.stp-enable", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.stp-enable failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_float_newf(HAL_IN, &(hal_data->vel_cmd), comp_id, "%s.%s.%s.stp-velo-cmd", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.stp-velo-cmd failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + + // export parameters + if ((err = hal_param_float_newf(HAL_RO, &(hal_data->freq), comp_id, "%s.%s.%s.stp-freq", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.stp-freq failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_param_float_newf(HAL_RO, &(hal_data->maxvel), comp_id, "%s.%s.%s.stp-maxvel", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.stp-maxvel failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_param_float_newf(HAL_RO, &(hal_data->maxaccel_fall), comp_id, "%s.%s.%s.stp-maxaccel-fall", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.stp-maxaccel-fall failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_param_float_newf(HAL_RO, &(hal_data->maxaccel_rise), comp_id, "%s.%s.%s.stp-maxaccel-rise", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.stp-maxaccel-rise\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_param_float_newf(HAL_RW, &(hal_data->pos_scale), comp_id, "%s.%s.%s.stp-pos-scale", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.stp-pos-scale failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + + // set default pin values + *(hal_data->count) = 0; + *(hal_data->pos_fb) = 0; + *(hal_data->ramp_active) = 0; + *(hal_data->ramp_disable) = 0; + *(hal_data->in_z) = 0; + *(hal_data->in_z_not) = 0; + *(hal_data->in_t) = 0; + *(hal_data->in_t_not) = 0; + *(hal_data->enable) = 0; + *(hal_data->vel_cmd) = 0.0; + + // init parameters + hal_data->freq = 0.0; + hal_data->pos_scale = 1.0; + hal_data->old_scale = 0.0; + hal_data->scale_recip = 0.0; + hal_data->maxvel = 0; + hal_data->maxaccel_rise = 0; + hal_data->maxaccel_fall = 0; + + // init other fields + hal_data->last_operational = 0; + hal_data->last_hw_count = 0; + + // calculate frequency factor + if (hal_data->sdo_base_freq != 0) { + hal_data->freqscale = (double)0x7fff / (double)hal_data->sdo_base_freq; + hal_data->freqscale_recip = 1 / hal_data->freqscale; + } else { + hal_data->freqscale = 0; + hal_data->freqscale_recip = 0; + } + + // calculate max frequency + if (hal_data->sdo_max_freq != 0) { + hal_data->max_freq = (double)hal_data->sdo_max_freq * hal_data->freqscale_recip; + } else { + hal_data->max_freq = (double)(hal_data->sdo_base_freq); + } + + // calculate maximum acceleartions in Hz/s + if ((hal_data->sdo_ramp_factor & 0x01) != 0) { + ramp_factor = 1000; + } else { + ramp_factor = 10; + } + hal_data->max_ac_rise = ramp_factor * (double)(hal_data->sdo_ramp_rise); + hal_data->max_ac_fall = ramp_factor * (double)(hal_data->sdo_ramp_fall); + + return 0; +} + +void lcec_el2521_check_scale(lcec_el2521_data_t *hal_data) { + // check for change in scale value + if (hal_data->pos_scale != hal_data->old_scale) { + // validate the new scale value + if ((hal_data->pos_scale < 1e-20) && (hal_data->pos_scale > -1e-20)) { + // value too small, divide by zero is a bad thing + hal_data->pos_scale = 1.0; + } + // get ready to detect future scale changes + hal_data->old_scale = hal_data->pos_scale; + // we will need the reciprocal + hal_data->scale_recip = 1.0 / hal_data->pos_scale; + } +} + +void lcec_el2521_read(struct lcec_slave *slave, long period) { + lcec_master_t *master = slave->master; + lcec_el2521_data_t *hal_data = (lcec_el2521_data_t *) slave->hal_data; + uint8_t *pd = master->process_data; + int16_t hw_count, hw_count_diff; + uint16_t state; + int in; + + // wait for slave to be operational + if (!slave->state.operational) { + hal_data->last_operational = 0; + return; + } + + // check for change in scale value + lcec_el2521_check_scale(hal_data); + + // calculate scaled limits + hal_data->maxvel = hal_data->max_freq * hal_data->scale_recip; + hal_data->maxaccel_rise = hal_data->max_ac_rise * hal_data->scale_recip; + hal_data->maxaccel_fall = hal_data->max_ac_fall * hal_data->scale_recip; + + // read state word + state = EC_READ_U16(&pd[hal_data->state_pdo_os]); + *(hal_data->ramp_active) = (state >> 1) & 1; + in = (state >> 5) & 1; + *(hal_data->in_z) = in; + *(hal_data->in_z_not) = !in; + in = (state >> 4) & 1; + *(hal_data->in_t) = in; + *(hal_data->in_t_not) = !in; + + // get counter diff + hw_count = EC_READ_S16(&pd[hal_data->count_pdo_os]); + hw_count_diff = hw_count - hal_data->last_hw_count; + hal_data->last_hw_count = hw_count; + if (!hal_data->last_operational) { + hw_count_diff = 0; + } + + // update raw count + *(hal_data->count) += hw_count_diff; + + // scale position + *(hal_data->pos_fb) = (double) (*(hal_data->count)) * hal_data->scale_recip; + + hal_data->last_operational = 1; +} + +void lcec_el2521_write(struct lcec_slave *slave, long period) { + lcec_master_t *master = slave->master; + lcec_el2521_data_t *hal_data = (lcec_el2521_data_t *) slave->hal_data; + uint8_t *pd = master->process_data; + uint16_t ctrl; + int32_t freq_raw; + + // check for change in scale value + lcec_el2521_check_scale(hal_data); + + // write control word + ctrl = 0; + if (*(hal_data->ramp_disable)) { + ctrl |= (1 << 1); + } + EC_WRITE_S16(&pd[hal_data->ctrl_pdo_os], ctrl); + + // update frequency + if (*(hal_data->enable)) { + hal_data->freq = *(hal_data->vel_cmd) * hal_data->pos_scale; + } else { + hal_data->freq = 0; + } + + // output frequency + freq_raw = hal_data->freq * hal_data->freqscale; + if (freq_raw > 0x7fff) { + freq_raw = 0x7fff; + } + if (freq_raw < -0x7fff) { + freq_raw = -0x7fff; + } + EC_WRITE_S16(&pd[hal_data->freq_pdo_os], freq_raw); +} + diff --git a/src/lcec_el2521.h b/src/lcec_el2521.h new file mode 100644 index 00000000..d2dcf10c --- /dev/null +++ b/src/lcec_el2521.h @@ -0,0 +1,39 @@ +// +// Copyright (C) 2011 Sascha Ittner +// +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2 of the License, or +// (at your option) any later version. +// +// This program 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 this program; if not, write to the Free Software +// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA +// +#ifndef _LCEC_EL2521_H_ +#define _LCEC_EL2521_H_ + +// **************************************************************************** +// CONFIG ISSUES: +// - sign/amount representation (8000:04) must be FALSE (0x00, default) +// - ramp function (8000:06) need to be active (0x01, default), can be disabled by hal pin +// - direct input mode (8000:08) must be FALSE (0x00, default) +// - travel distance control active (8000:0A) must be FALSE (0x00, default) +// **************************************************************************** + +#include "lcec.h" + +#define LCEC_EL2521_VID LCEC_BECKHOFF_VID +#define LCEC_EL2521_PID 0x09d93052 + +#define LCEC_EL2521_PDOS 4 + +int lcec_el2521_init(int comp_id, struct lcec_slave *slave, ec_pdo_entry_reg_t *pdo_entry_regs); + +#endif + diff --git a/src/lcec_el2xxx.c b/src/lcec_el2xxx.c new file mode 100644 index 00000000..45c722c7 --- /dev/null +++ b/src/lcec_el2xxx.c @@ -0,0 +1,88 @@ +// +// Copyright (C) 2011 Sascha Ittner +// +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2 of the License, or +// (at your option) any later version. +// +// This program 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 this program; if not, write to the Free Software +// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA +// + +#include "lcec.h" +#include "lcec_el2xxx.h" + +typedef struct { + hal_bit_t *out; + hal_bit_t invert; + unsigned int pdo_os; + unsigned int pdo_bp; +} lcec_el2xxx_pin_t; + +void lcec_el2xxx_write(struct lcec_slave *slave, long period); + +int lcec_el2xxx_init(int comp_id, struct lcec_slave *slave, ec_pdo_entry_reg_t *pdo_entry_regs) { + lcec_master_t *master = slave->master; + lcec_el2xxx_pin_t *hal_data; + lcec_el2xxx_pin_t *pin; + int i; + int err; + + // initialize callbacks + slave->proc_write = lcec_el2xxx_write; + + // alloc hal memory + if ((hal_data = hal_malloc(sizeof(lcec_el2xxx_pin_t) * slave->pdo_entry_count)) == NULL) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "hal_malloc() for slave %s.%s failed\n", master->name, slave->name); + return -EIO; + } + memset(hal_data, 0, sizeof(lcec_el2xxx_pin_t) * slave->pdo_entry_count); + slave->hal_data = hal_data; + + // initialize pins + for (i=0, pin=hal_data; ipdo_entry_count; i++, pin++) { + // initialize POD entry + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x7000 + (i << 4), 0x01, &pin->pdo_os, &pin->pdo_bp); + + // export pins + if ((err = hal_pin_bit_newf(HAL_IN, &(pin->out), comp_id, "%s.%s.%s.dout-%d", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.dout-%02d failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_param_bit_newf(HAL_RW, &(pin->invert), comp_id, "%s.%s.%s.dout-%d-invert", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.dout-%02d-invert failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + + // initialize pins + *(pin->out) = 0; + pin->invert = 0; + } + + return 0; +} + +void lcec_el2xxx_write(struct lcec_slave *slave, long period) { + lcec_master_t *master = slave->master; + lcec_el2xxx_pin_t *hal_data = (lcec_el2xxx_pin_t *) slave->hal_data; + uint8_t *pd = master->process_data; + lcec_el2xxx_pin_t *pin; + int i, s; + + // set outputs + for (i=0, pin=hal_data; ipdo_entry_count; i++, pin++) { + s = *(pin->out); + if (pin->invert) { + s = !s; + } + EC_WRITE_BIT(&pd[pin->pdo_os], pin->pdo_bp, s); + } +} + diff --git a/src/lcec_el2xxx.h b/src/lcec_el2xxx.h new file mode 100644 index 00000000..6e0a7c63 --- /dev/null +++ b/src/lcec_el2xxx.h @@ -0,0 +1,56 @@ +// +// Copyright (C) 2011 Sascha Ittner +// +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2 of the License, or +// (at your option) any later version. +// +// This program 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 this program; if not, write to the Free Software +// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA +// +#ifndef _LCEC_EL2XXX_H_ +#define _LCEC_EL2XXX_H_ + +#include "lcec.h" + +#define LCEC_EL2xxx_VID LCEC_BECKHOFF_VID + +#define LCEC_EL2002_PID 0x07D23052 +#define LCEC_EL2004_PID 0x07D43052 +#define LCEC_EL2008_PID 0x07D83052 +#define LCEC_EL2022_PID 0x07E63052 +#define LCEC_EL2024_PID 0x07E83052 +#define LCEC_EL2032_PID 0x07F03052 +#define LCEC_EL2034_PID 0x07F23052 +#define LCEC_EL2042_PID 0x07FA3052 +#define LCEC_EL2084_PID 0x08243052 +#define LCEC_EL2088_PID 0x08283052 +#define LCEC_EL2124_PID 0x084C3052 +#define LCEC_EL2808_PID 0x0AF83052 +#define LCEC_EL2809_PID 0x0AF93052 + +#define LCEC_EL2002_PDOS 2 +#define LCEC_EL2004_PDOS 4 +#define LCEC_EL2008_PDOS 8 +#define LCEC_EL2022_PDOS 2 +#define LCEC_EL2024_PDOS 4 +#define LCEC_EL2032_PDOS 2 +#define LCEC_EL2034_PDOS 4 +#define LCEC_EL2042_PDOS 2 +#define LCEC_EL2084_PDOS 4 +#define LCEC_EL2088_PDOS 8 +#define LCEC_EL2124_PDOS 4 +#define LCEC_EL2808_PDOS 8 +#define LCEC_EL2809_PDOS 16 + +int lcec_el2xxx_init(int comp_id, struct lcec_slave *slave, ec_pdo_entry_reg_t *pdo_entry_regs); + +#endif + diff --git a/src/lcec_el31x2.c b/src/lcec_el31x2.c new file mode 100644 index 00000000..baad7ac3 --- /dev/null +++ b/src/lcec_el31x2.c @@ -0,0 +1,165 @@ +// +// Copyright (C) 2011 Sascha Ittner +// +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2 of the License, or +// (at your option) any later version. +// +// This program 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 this program; if not, write to the Free Software +// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA +// + +#include "lcec.h" +#include "lcec_el31x2.h" + +typedef struct { + hal_bit_t *error; + hal_bit_t *overrange; + hal_bit_t *underrange; + hal_s32_t *raw_val; + hal_float_t *scale; + hal_float_t *bias; + hal_float_t *val; + unsigned int state_pdo_os; + unsigned int val_pdo_os; +} lcec_el31x2_chan_t; + +typedef struct { + lcec_el31x2_chan_t chans[LCEC_EL31x2_CHANS]; +} lcec_el31x2_data_t; + +static ec_pdo_entry_info_t lcec_el31x2_channel1[] = { + {0x3101, 1, 8}, // status + {0x3101, 2, 16} // value +}; + +static ec_pdo_entry_info_t lcec_el31x2_channel2[] = { + {0x3102, 1, 8}, // status + {0x3102, 2, 16} // value +}; + +static ec_pdo_info_t lcec_el31x2_pdos_in[] = { + {0x1A00, 2, lcec_el31x2_channel1}, + {0x1A01, 2, lcec_el31x2_channel2} +}; + +static ec_sync_info_t lcec_el31x2_syncs[] = { + {0, EC_DIR_OUTPUT, 0, NULL}, + {1, EC_DIR_INPUT, 0, NULL}, + {2, EC_DIR_OUTPUT, 0, NULL}, + {3, EC_DIR_INPUT, 2, lcec_el31x2_pdos_in}, + {0xff} +}; + +void lcec_el31x2_read(struct lcec_slave *slave, long period); + +int lcec_el31x2_init(int comp_id, struct lcec_slave *slave, ec_pdo_entry_reg_t *pdo_entry_regs) { + lcec_master_t *master = slave->master; + lcec_el31x2_data_t *hal_data; + lcec_el31x2_chan_t *chan; + int i; + int err; + + // initialize callbacks + slave->proc_read = lcec_el31x2_read; + + // alloc hal memory + if ((hal_data = hal_malloc(sizeof(lcec_el31x2_data_t))) == NULL) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "hal_malloc() for slave %s.%s failed\n", master->name, slave->name); + return -EIO; + } + memset(hal_data, 0, sizeof(lcec_el31x2_data_t)); + slave->hal_data = hal_data; + + // initializer sync info + slave->sync_info = lcec_el31x2_syncs; + + // initialize pins + for (i=0; ichans[i]; + + // initialize POD entries + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x3101 + i, 0x01, &chan->state_pdo_os, NULL); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x3101 + i, 0x02, &chan->val_pdo_os, NULL); + + // export pins + if ((err = hal_pin_bit_newf(HAL_OUT, &(chan->error), comp_id, "%s.%s.%s.ain-%d-error", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.ain-%d-error failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(chan->overrange), comp_id, "%s.%s.%s.ain-%d-overrange", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.ain-%d-overrange failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(chan->underrange), comp_id, "%s.%s.%s.ain-%d-underrange", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.ain-%d-underrange failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_s32_newf(HAL_OUT, &(chan->raw_val), comp_id, "%s.%s.%s.ain-%d-raw", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.ain-%d-raw failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_float_newf(HAL_OUT, &(chan->val), comp_id, "%s.%s.%s.ain-%d-val", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.ain-%d-val failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_float_newf(HAL_IO, &(chan->scale), comp_id, "%s.%s.%s.ain-%d-scale", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.ain-%d-scale failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_float_newf(HAL_IO, &(chan->bias), comp_id, "%s.%s.%s.ain-%d-bias", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.ain-%d-bias failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + + // initialize pins + *(chan->error) = 0; + *(chan->overrange) = 0; + *(chan->underrange) = 0; + *(chan->raw_val) = 0; + *(chan->val) = 0; + *(chan->scale) = 1; + *(chan->bias) = 0; + } + + return 0; +} + +void lcec_el31x2_read(struct lcec_slave *slave, long period) { + lcec_master_t *master = slave->master; + lcec_el31x2_data_t *hal_data = (lcec_el31x2_data_t *) slave->hal_data; + uint8_t *pd = master->process_data; + int i; + lcec_el31x2_chan_t *chan; + uint8_t state; + int16_t value; + + // wait for slave to be operational + if (!slave->state.operational) { + return; + } + + // check inputs + for (i=0; ichans[i]; + + // update state + state = pd[chan->state_pdo_os]; + *(chan->error) = (state >> 6) & 0x01; + *(chan->overrange) = (state >> 1) & 0x01; + *(chan->underrange) = (state >> 0) & 0x01; + + // update value + value = EC_READ_S16(&pd[chan->val_pdo_os]); + *(chan->raw_val) = value; + *(chan->val) = *(chan->bias) + *(chan->scale) * (double)value * ((double)1/(double)0x7fff); + } +} + diff --git a/src/lcec_el31x2.h b/src/lcec_el31x2.h new file mode 100644 index 00000000..93a4b525 --- /dev/null +++ b/src/lcec_el31x2.h @@ -0,0 +1,39 @@ +// +// Copyright (C) 2011 Sascha Ittner +// +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2 of the License, or +// (at your option) any later version. +// +// This program 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 this program; if not, write to the Free Software +// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA +// +#ifndef _LCEC_EL31X2_H_ +#define _LCEC_EL31X2_H_ + +#include "lcec.h" + +#define LCEC_EL31x2_VID LCEC_BECKHOFF_VID + +#define LCEC_EL3102_PID 0x0C1E3052 +#define LCEC_EL3112_PID 0x0C283052 +#define LCEC_EL3122_PID 0x0C323052 +#define LCEC_EL3142_PID 0x0C463052 +#define LCEC_EL3152_PID 0x0C503052 +#define LCEC_EL3162_PID 0x0C5A3052 + +#define LCEC_EL31x2_PDOS 4 + +#define LCEC_EL31x2_CHANS 2 + +int lcec_el31x2_init(int comp_id, struct lcec_slave *slave, ec_pdo_entry_reg_t *pdo_entry_regs); + +#endif + diff --git a/src/lcec_el40x1.c b/src/lcec_el40x1.c new file mode 100644 index 00000000..b3967488 --- /dev/null +++ b/src/lcec_el40x1.c @@ -0,0 +1,220 @@ +// +// Copyright (C) 2011 Sascha Ittner +// +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2 of the License, or +// (at your option) any later version. +// +// This program 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 this program; if not, write to the Free Software +// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA +// + +#include "lcec.h" +#include "lcec_el40x1.h" + +typedef struct { + hal_bit_t *pos; + hal_bit_t *neg; + hal_bit_t *enable; + hal_bit_t *absmode; + hal_float_t *value; + hal_float_t *scale; + hal_float_t *offset; + double old_scale; + double scale_recip; + hal_float_t *min_dc; + hal_float_t *max_dc; + hal_float_t *curr_dc; + hal_s32_t *raw_val; + unsigned int val_pdo_os; +} lcec_el40x1_data_t; + +static ec_pdo_entry_info_t lcec_el40x1_channel[] = { + {0x7000, 1, 16} // output +}; + +static ec_pdo_info_t lcec_el40x1_pdos_in[] = { + {0x1600, 1, lcec_el40x1_channel}, +}; + +static ec_sync_info_t lcec_el40x1_syncs[] = { + {0, EC_DIR_OUTPUT, 0, NULL}, + {1, EC_DIR_INPUT, 0, NULL}, + {2, EC_DIR_OUTPUT, 1, lcec_el40x1_pdos_in}, + {0xff} +}; + +void lcec_el40x1_write(struct lcec_slave *slave, long period); + +int lcec_el40x1_init(int comp_id, struct lcec_slave *slave, ec_pdo_entry_reg_t *pdo_entry_regs) { + lcec_master_t *master = slave->master; + lcec_el40x1_data_t *hal_data; + int err; + + // initialize callbacks + slave->proc_write = lcec_el40x1_write; + + // alloc hal memory + if ((hal_data = hal_malloc(sizeof(lcec_el40x1_data_t))) == NULL) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "hal_malloc() for slave %s.%s failed\n", master->name, slave->name); + return -EIO; + } + memset(hal_data, 0, sizeof(lcec_el40x1_data_t)); + slave->hal_data = hal_data; + + // initializer sync info + slave->sync_info = lcec_el40x1_syncs; + + // initialize POD entries + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x7000, 0x01, &hal_data->val_pdo_os, NULL); + + // export paramameters + if ((err = hal_pin_float_newf(HAL_IO, &(hal_data->scale), comp_id, "%s.%s.%s.aout-scale", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.aout-scale failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_float_newf(HAL_IO, &(hal_data->offset), comp_id, "%s.%s.%s.aout-offset", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.aout-offset failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_float_newf(HAL_IO, &(hal_data->min_dc), comp_id, "%s.%s.%s.aout-min-dc", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.aout-min-dc failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_float_newf(HAL_IO, &(hal_data->max_dc), comp_id, "%s.%s.%s.aout-max-dc", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.aout-max-dc failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_float_newf(HAL_OUT, &(hal_data->curr_dc), comp_id, "%s.%s.%s.aout-curr-dc", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.aout-curr-dc failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + + // export pins + if ((err = hal_pin_bit_newf(HAL_IN, &(hal_data->enable), comp_id, "%s.%s.%s.aout-enable", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.aout-enable failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_IN, &(hal_data->absmode), comp_id, "%s.%s.%s.aout-absmode", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.aout-absmode failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_float_newf(HAL_IN, &(hal_data->value), comp_id, "%s.%s.%s.aout-value", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.aout-value failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_s32_newf(HAL_OUT, &(hal_data->raw_val), comp_id, "%s.%s.%s.aout-raw", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.aout-raw failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + + // export UP/DOWN pins + if ((err = hal_pin_bit_newf(HAL_OUT, &(hal_data->pos), comp_id, "%s.%s.%s.aout-pos", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.aout-pos failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(hal_data->neg), comp_id, "%s.%s.%s.aout-neg", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.aout-neg failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + + // set default pin values + *(hal_data->scale) = 1.0; + *(hal_data->offset) = 0.0; + *(hal_data->min_dc) = -1.0; + *(hal_data->max_dc) = 1.0; + *(hal_data->curr_dc) = 0.0; + *(hal_data->enable) = 0; + *(hal_data->absmode) = 0; + *(hal_data->value) = 0.0; + *(hal_data->raw_val) = 0.0; + *(hal_data->pos) = 0; + *(hal_data->neg) = 0; + + // init other fields + hal_data->old_scale = *(hal_data->scale) + 1.0; + hal_data->scale_recip = 1.0; + + return 0; +} + +void lcec_el40x1_write(struct lcec_slave *slave, long period) { + lcec_master_t *master = slave->master; + lcec_el40x1_data_t *hal_data = (lcec_el40x1_data_t *) slave->hal_data; + uint8_t *pd = master->process_data; + double tmpval, tmpdc, raw_val; + + // validate duty cycle limits, both limits must be between + // 0.0 and 1.0 (inclusive) and max must be greater then min + if (*(hal_data->max_dc) > 1.0) { + *(hal_data->max_dc) = 1.0; + } + if (*(hal_data->min_dc) > *(hal_data->max_dc)) { + *(hal_data->min_dc) = *(hal_data->max_dc); + } + if (*(hal_data->min_dc) < -1.0) { + *(hal_data->min_dc) = -1.0; + } + if (*(hal_data->max_dc) < *(hal_data->min_dc)) { + *(hal_data->max_dc) = *(hal_data->min_dc); + } + + // do scale calcs only when scale changes + if (*(hal_data->scale) != hal_data->old_scale) { + // validate the new scale value + if ((*(hal_data->scale) < 1e-20) && (*(hal_data->scale) > -1e-20)) { + // value too small, divide by zero is a bad thing + *(hal_data->scale) = 1.0; + } + // get ready to detect future scale changes + hal_data->old_scale = *(hal_data->scale); + // we will need the reciprocal + hal_data->scale_recip = 1.0 / *(hal_data->scale); + } + + // get command + tmpval = *(hal_data->value); + if (*(hal_data->absmode) && (tmpval < 0)) { + tmpval = -tmpval; + } + + // convert value command to duty cycle + tmpdc = tmpval * hal_data->scale_recip + *(hal_data->offset); + if (tmpdc < *(hal_data->min_dc)) { + tmpdc = *(hal_data->min_dc); + } + if (tmpdc > *(hal_data->max_dc)) { + tmpdc = *(hal_data->max_dc); + } + + // set output values + if (*(hal_data->enable) == 0) { + raw_val = 0; + *(hal_data->pos) = 0; + *(hal_data->neg) = 0; + *(hal_data->curr_dc) = 0; + } else { + raw_val = (double)0x7fff * tmpdc; + if (raw_val > (double)0x7fff) { + raw_val = (double)0x7fff; + } + if (raw_val < (double)-0x7fff) { + raw_val = (double)-0x7fff; + } + *(hal_data->pos) = (*(hal_data->value) > 0); + *(hal_data->neg) = (*(hal_data->value) < 0); + *(hal_data->curr_dc) = tmpdc; + } + + // update value + EC_WRITE_S16(&pd[hal_data->val_pdo_os], (int16_t)raw_val); + *(hal_data->raw_val) = (int32_t)raw_val; +} + diff --git a/src/lcec_el40x1.h b/src/lcec_el40x1.h new file mode 100644 index 00000000..fb9092db --- /dev/null +++ b/src/lcec_el40x1.h @@ -0,0 +1,35 @@ +// +// Copyright (C) 2011 Sascha Ittner +// +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2 of the License, or +// (at your option) any later version. +// +// This program 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 this program; if not, write to the Free Software +// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA +// +#ifndef _LCEC_EL40X1_H_ +#define _LCEC_EL40X1_H_ + +#include "lcec.h" + +#define LCEC_EL40x1_VID LCEC_BECKHOFF_VID + +#define LCEC_EL4001_PID 0x0fa13052 +#define LCEC_EL4011_PID 0x0fab3052 +#define LCEC_EL4021_PID 0x0fb53052 +#define LCEC_EL4031_PID 0x0fbf3052 + +#define LCEC_EL40x1_PDOS 1 + +int lcec_el40x1_init(int comp_id, struct lcec_slave *slave, ec_pdo_entry_reg_t *pdo_entry_regs); + +#endif + diff --git a/src/lcec_el40x2.c b/src/lcec_el40x2.c new file mode 100644 index 00000000..0be8ffaa --- /dev/null +++ b/src/lcec_el40x2.c @@ -0,0 +1,243 @@ +// +// Copyright (C) 2011 Sascha Ittner +// +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2 of the License, or +// (at your option) any later version. +// +// This program 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 this program; if not, write to the Free Software +// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA +// + +#include "lcec.h" +#include "lcec_el40x2.h" + +typedef struct { + hal_bit_t *pos; + hal_bit_t *neg; + hal_bit_t *enable; + hal_bit_t *absmode; + hal_float_t *value; + hal_float_t *scale; + hal_float_t *offset; + double old_scale; + double scale_recip; + hal_float_t *min_dc; + hal_float_t *max_dc; + hal_float_t *curr_dc; + hal_s32_t *raw_val; + unsigned int val_pdo_os; +} lcec_el40x2_chan_t; + +typedef struct { + lcec_el40x2_chan_t chans[LCEC_EL40x2_CHANS]; +} lcec_el40x2_data_t; + +static ec_pdo_entry_info_t lcec_el40x2_channel1[] = { + {0x7000, 1, 16} // output +}; + +static ec_pdo_entry_info_t lcec_el40x2_channel2[] = { + {0x7010, 1, 16} // output +}; + +static ec_pdo_info_t lcec_el40x2_pdos_in[] = { + {0x1600, 1, lcec_el40x2_channel1}, + {0x1601, 1, lcec_el40x2_channel2} +}; + +static ec_sync_info_t lcec_el40x2_syncs[] = { + {0, EC_DIR_OUTPUT, 0, NULL}, + {1, EC_DIR_INPUT, 0, NULL}, + {2, EC_DIR_OUTPUT, 2, lcec_el40x2_pdos_in}, + {0xff} +}; + +void lcec_el40x2_write(struct lcec_slave *slave, long period); + +int lcec_el40x2_init(int comp_id, struct lcec_slave *slave, ec_pdo_entry_reg_t *pdo_entry_regs) { + lcec_master_t *master = slave->master; + lcec_el40x2_data_t *hal_data; + lcec_el40x2_chan_t *chan; + int i; + int err; + + // initialize callbacks + slave->proc_write = lcec_el40x2_write; + + // alloc hal memory + if ((hal_data = hal_malloc(sizeof(lcec_el40x2_data_t))) == NULL) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "hal_malloc() for slave %s.%s failed\n", master->name, slave->name); + return -EIO; + } + memset(hal_data, 0, sizeof(lcec_el40x2_data_t)); + slave->hal_data = hal_data; + + // initializer sync info + slave->sync_info = lcec_el40x2_syncs; + + // initialize pins + for (i=0; ichans[i]; + + // initialize POD entries + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x7000 + (i << 4), 0x01, &chan->val_pdo_os, NULL); + + // export paramameters + if ((err = hal_pin_float_newf(HAL_IO, &(chan->scale), comp_id, "%s.%s.%s.aout-%d-scale", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.aout-%d-scale failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_float_newf(HAL_IO, &(chan->offset), comp_id, "%s.%s.%s.aout-%d-offset", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.aout-%d-offset failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_float_newf(HAL_IO, &(chan->min_dc), comp_id, "%s.%s.%s.aout-%d-min-dc", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.aout-%d-min-dc failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_float_newf(HAL_IO, &(chan->max_dc), comp_id, "%s.%s.%s.aout-%d-max-dc", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.aout-%d-max-dc failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_float_newf(HAL_OUT, &(chan->curr_dc), comp_id, "%s.%s.%s.aout-%d-curr-dc", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.aout-%d-curr-dc failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + + // export pins + if ((err = hal_pin_bit_newf(HAL_IN, &(chan->enable), comp_id, "%s.%s.%s.aout-%d-enable", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.aout-%d-enable failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_IN, &(chan->absmode), comp_id, "%s.%s.%s.aout-%d-absmode", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.aout-%d-absmode failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_float_newf(HAL_IN, &(chan->value), comp_id, "%s.%s.%s.aout-%d-value", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.aout-%d-value failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_s32_newf(HAL_OUT, &(chan->raw_val), comp_id, "%s.%s.%s.aout-%d-raw", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.aout-%d-raw failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + + // export UP/DOWN pins + if ((err = hal_pin_bit_newf(HAL_OUT, &(chan->pos), comp_id, "%s.%s.%s.aout-%d-pos", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.aout-%d-pos failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(chan->neg), comp_id, "%s.%s.%s.aout-%d-neg", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.aout-%d-neg failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + + // set default pin values + *(chan->scale) = 1.0; + *(chan->offset) = 0.0; + *(chan->min_dc) = -1.0; + *(chan->max_dc) = 1.0; + *(chan->curr_dc) = 0.0; + *(chan->enable) = 0; + *(chan->absmode) = 0; + *(chan->value) = 0.0; + *(chan->raw_val) = 0.0; + *(chan->pos) = 0; + *(chan->neg) = 0; + + // init other fields + chan->old_scale = *(chan->scale) + 1.0; + chan->scale_recip = 1.0; + } + + return 0; +} + +void lcec_el40x2_write(struct lcec_slave *slave, long period) { + lcec_master_t *master = slave->master; + lcec_el40x2_data_t *hal_data = (lcec_el40x2_data_t *) slave->hal_data; + uint8_t *pd = master->process_data; + int i; + lcec_el40x2_chan_t *chan; + double tmpval, tmpdc, raw_val; + + // check inputs + for (i=0; ichans[i]; + + // validate duty cycle limits, both limits must be between + // 0.0 and 1.0 (inclusive) and max must be greater then min + if (*(chan->max_dc) > 1.0) { + *(chan->max_dc) = 1.0; + } + if (*(chan->min_dc) > *(chan->max_dc)) { + *(chan->min_dc) = *(chan->max_dc); + } + if (*(chan->min_dc) < -1.0) { + *(chan->min_dc) = -1.0; + } + if (*(chan->max_dc) < *(chan->min_dc)) { + *(chan->max_dc) = *(chan->min_dc); + } + + // do scale calcs only when scale changes + if (*(chan->scale) != chan->old_scale) { + // validate the new scale value + if ((*(chan->scale) < 1e-20) && (*(chan->scale) > -1e-20)) { + // value too small, divide by zero is a bad thing + *(chan->scale) = 1.0; + } + // get ready to detect future scale changes + chan->old_scale = *(chan->scale); + // we will need the reciprocal + chan->scale_recip = 1.0 / *(chan->scale); + } + + // get command + tmpval = *(chan->value); + if (*(chan->absmode) && (tmpval < 0)) { + tmpval = -tmpval; + } + + // convert value command to duty cycle + tmpdc = tmpval * chan->scale_recip + *(chan->offset); + if (tmpdc < *(chan->min_dc)) { + tmpdc = *(chan->min_dc); + } + if (tmpdc > *(chan->max_dc)) { + tmpdc = *(chan->max_dc); + } + + // set output values + if (*(chan->enable) == 0) { + raw_val = 0; + *(chan->pos) = 0; + *(chan->neg) = 0; + *(chan->curr_dc) = 0; + } else { + raw_val = (double)0x7fff * tmpdc; + if (raw_val > (double)0x7fff) { + raw_val = (double)0x7fff; + } + if (raw_val < (double)-0x7fff) { + raw_val = (double)-0x7fff; + } + *(chan->pos) = (*(chan->value) > 0); + *(chan->neg) = (*(chan->value) < 0); + *(chan->curr_dc) = tmpdc; + } + + // update value + EC_WRITE_S16(&pd[chan->val_pdo_os], (int16_t)raw_val); + *(chan->raw_val) = (int32_t)raw_val; + } +} + diff --git a/src/lcec_el40x2.h b/src/lcec_el40x2.h new file mode 100644 index 00000000..4930af06 --- /dev/null +++ b/src/lcec_el40x2.h @@ -0,0 +1,37 @@ +// +// Copyright (C) 2011 Sascha Ittner +// +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2 of the License, or +// (at your option) any later version. +// +// This program 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 this program; if not, write to the Free Software +// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA +// +#ifndef _LCEC_EL40X2_H_ +#define _LCEC_EL40X2_H_ + +#include "lcec.h" + +#define LCEC_EL40x2_VID LCEC_BECKHOFF_VID + +#define LCEC_EL4002_PID 0x0fa23052 +#define LCEC_EL4012_PID 0x0fac3052 +#define LCEC_EL4022_PID 0x0fb63052 +#define LCEC_EL4032_PID 0x0fc03052 + +#define LCEC_EL40x2_PDOS 2 + +#define LCEC_EL40x2_CHANS 2 + +int lcec_el40x2_init(int comp_id, struct lcec_slave *slave, ec_pdo_entry_reg_t *pdo_entry_regs); + +#endif + diff --git a/src/lcec_el41x2.c b/src/lcec_el41x2.c new file mode 100644 index 00000000..abc94f5a --- /dev/null +++ b/src/lcec_el41x2.c @@ -0,0 +1,243 @@ +// +// Copyright (C) 2011 Sascha Ittner +// +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2 of the License, or +// (at your option) any later version. +// +// This program 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 this program; if not, write to the Free Software +// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA +// + +#include "lcec.h" +#include "lcec_el41x2.h" + +typedef struct { + hal_bit_t *pos; + hal_bit_t *neg; + hal_bit_t *enable; + hal_bit_t *absmode; + hal_float_t *value; + hal_float_t *scale; + hal_float_t *offset; + double old_scale; + double scale_recip; + hal_float_t *min_dc; + hal_float_t *max_dc; + hal_float_t *curr_dc; + hal_s32_t *raw_val; + unsigned int val_pdo_os; +} lcec_el41x2_chan_t; + +typedef struct { + lcec_el41x2_chan_t chans[LCEC_EL41x2_CHANS]; +} lcec_el41x2_data_t; + +static ec_pdo_entry_info_t lcec_el41x2_channel1[] = { + {0x3001, 1, 16} // output +}; + +static ec_pdo_entry_info_t lcec_el41x2_channel2[] = { + {0x3002, 1, 16} // output +}; + +static ec_pdo_info_t lcec_el41x2_pdos_in[] = { + {0x1600, 1, lcec_el41x2_channel1}, + {0x1601, 1, lcec_el41x2_channel2} +}; + +static ec_sync_info_t lcec_el41x2_syncs[] = { + {0, EC_DIR_OUTPUT, 0, NULL}, + {1, EC_DIR_INPUT, 0, NULL}, + {2, EC_DIR_OUTPUT, 2, lcec_el41x2_pdos_in}, + {0xff} +}; + +void lcec_el41x2_write(struct lcec_slave *slave, long period); + +int lcec_el41x2_init(int comp_id, struct lcec_slave *slave, ec_pdo_entry_reg_t *pdo_entry_regs) { + lcec_master_t *master = slave->master; + lcec_el41x2_data_t *hal_data; + lcec_el41x2_chan_t *chan; + int i; + int err; + + // initialize callbacks + slave->proc_write = lcec_el41x2_write; + + // alloc hal memory + if ((hal_data = hal_malloc(sizeof(lcec_el41x2_data_t))) == NULL) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "hal_malloc() for slave %s.%s failed\n", master->name, slave->name); + return -EIO; + } + memset(hal_data, 0, sizeof(lcec_el41x2_data_t)); + slave->hal_data = hal_data; + + // initializer sync info + slave->sync_info = lcec_el41x2_syncs; + + // initialize pins + for (i=0; ichans[i]; + + // initialize POD entries + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x3001 + i, 0x01, &chan->val_pdo_os, NULL); + + // export paramameters + if ((err = hal_pin_float_newf(HAL_IO, &(chan->scale), comp_id, "%s.%s.%s.aout-%d-scale", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.aout-%d-scale failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_float_newf(HAL_IO, &(chan->offset), comp_id, "%s.%s.%s.aout-%d-offset", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.aout-%d-offset failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_float_newf(HAL_IO, &(chan->min_dc), comp_id, "%s.%s.%s.aout-%d-min-dc", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.aout-%d-min-dc failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_float_newf(HAL_IO, &(chan->max_dc), comp_id, "%s.%s.%s.aout-%d-max-dc", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.aout-%d-max-dc failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_float_newf(HAL_OUT, &(chan->curr_dc), comp_id, "%s.%s.%s.aout-%d-curr-dc", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.aout-%d-curr-dc failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + + // export pins + if ((err = hal_pin_bit_newf(HAL_IN, &(chan->enable), comp_id, "%s.%s.%s.aout-%d-enable", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.aout-%d-enable failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_IN, &(chan->absmode), comp_id, "%s.%s.%s.aout-%d-absmode", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.aout-%d-absmode failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_float_newf(HAL_IN, &(chan->value), comp_id, "%s.%s.%s.aout-%d-value", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.aout-%d-value failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_s32_newf(HAL_OUT, &(chan->raw_val), comp_id, "%s.%s.%s.aout-%d-raw", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.aout-%d-raw failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + + // export UP/DOWN pins + if ((err = hal_pin_bit_newf(HAL_OUT, &(chan->pos), comp_id, "%s.%s.%s.aout-%d-pos", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.aout-%d-pos failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(chan->neg), comp_id, "%s.%s.%s.aout-%d-neg", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.aout-%d-neg failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + + // set default pin values + *(chan->scale) = 1.0; + *(chan->offset) = 0.0; + *(chan->min_dc) = -1.0; + *(chan->max_dc) = 1.0; + *(chan->curr_dc) = 0.0; + *(chan->enable) = 0; + *(chan->absmode) = 0; + *(chan->value) = 0.0; + *(chan->raw_val) = 0.0; + *(chan->pos) = 0; + *(chan->neg) = 0; + + // init other fields + chan->old_scale = *(chan->scale) + 1.0; + chan->scale_recip = 1.0; + } + + return 0; +} + +void lcec_el41x2_write(struct lcec_slave *slave, long period) { + lcec_master_t *master = slave->master; + lcec_el41x2_data_t *hal_data = (lcec_el41x2_data_t *) slave->hal_data; + uint8_t *pd = master->process_data; + int i; + lcec_el41x2_chan_t *chan; + double tmpval, tmpdc, raw_val; + + // set outputs + for (i=0; ichans[i]; + + // validate duty cycle limits, both limits must be between + // 0.0 and 1.0 (inclusive) and max must be greater then min + if (*(chan->max_dc) > 1.0) { + *(chan->max_dc) = 1.0; + } + if (*(chan->min_dc) > *(chan->max_dc)) { + *(chan->min_dc) = *(chan->max_dc); + } + if (*(chan->min_dc) < -1.0) { + *(chan->min_dc) = -1.0; + } + if (*(chan->max_dc) < *(chan->min_dc)) { + *(chan->max_dc) = *(chan->min_dc); + } + + // do scale calcs only when scale changes + if (*(chan->scale) != chan->old_scale) { + // validate the new scale value + if ((*(chan->scale) < 1e-20) && (*(chan->scale) > -1e-20)) { + // value too small, divide by zero is a bad thing + *(chan->scale) = 1.0; + } + // get ready to detect future scale changes + chan->old_scale = *(chan->scale); + // we will need the reciprocal + chan->scale_recip = 1.0 / *(chan->scale); + } + + // get command + tmpval = *(chan->value); + if (*(chan->absmode) && (tmpval < 0)) { + tmpval = -tmpval; + } + + // convert value command to duty cycle + tmpdc = tmpval * chan->scale_recip + *(chan->offset); + if (tmpdc < *(chan->min_dc)) { + tmpdc = *(chan->min_dc); + } + if (tmpdc > *(chan->max_dc)) { + tmpdc = *(chan->max_dc); + } + + // set output values + if (*(chan->enable) == 0) { + raw_val = 0; + *(chan->pos) = 0; + *(chan->neg) = 0; + *(chan->curr_dc) = 0; + } else { + raw_val = (double)0x7fff * tmpdc; + if (raw_val > (double)0x7fff) { + raw_val = (double)0x7fff; + } + if (raw_val < (double)-0x7fff) { + raw_val = (double)-0x7fff; + } + *(chan->pos) = (*(chan->value) > 0); + *(chan->neg) = (*(chan->value) < 0); + *(chan->curr_dc) = tmpdc; + } + + // update value + EC_WRITE_S16(&pd[chan->val_pdo_os], (int16_t)raw_val); + *(chan->raw_val) = (int32_t)raw_val; + } +} + diff --git a/src/lcec_el41x2.h b/src/lcec_el41x2.h new file mode 100644 index 00000000..6768677c --- /dev/null +++ b/src/lcec_el41x2.h @@ -0,0 +1,37 @@ +// +// Copyright (C) 2011 Sascha Ittner +// +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2 of the License, or +// (at your option) any later version. +// +// This program 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 this program; if not, write to the Free Software +// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA +// +#ifndef _LCEC_EL41X2_H_ +#define _LCEC_EL41X2_H_ + +#include "lcec.h" + +#define LCEC_EL41x2_VID LCEC_BECKHOFF_VID + +#define LCEC_EL4102_PID 0x10063052 +#define LCEC_EL4112_PID 0x10103052 +#define LCEC_EL4122_PID 0x101A3052 +#define LCEC_EL4132_PID 0x10243052 + +#define LCEC_EL41x2_PDOS 2 + +#define LCEC_EL41x2_CHANS 2 + +int lcec_el41x2_init(int comp_id, struct lcec_slave *slave, ec_pdo_entry_reg_t *pdo_entry_regs); + +#endif + diff --git a/src/lcec_el5101.c b/src/lcec_el5101.c new file mode 100644 index 00000000..81fd4bf8 --- /dev/null +++ b/src/lcec_el5101.c @@ -0,0 +1,387 @@ +// +// Copyright (C) 2012 Sascha Ittner +// +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2 of the License, or +// (at your option) any later version. +// +// This program 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 this program; if not, write to the Free Software +// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA +// + +#include "lcec.h" +#include "lcec_el5101.h" + +#define LCEC_EL5101_STATUS_INPUT (1 << 5) +#define LCEC_EL5101_STATUS_OVERFLOW (1 << 4) +#define LCEC_EL5101_STATUS_UNDERFLOW (1 << 3) +#define LCEC_EL5101_STATUS_CNTSET_ACC (1 << 2) +#define LCEC_EL5101_STATUS_LAT_EXT_VAL (1 << 1) +#define LCEC_EL5101_STATUS_LATC_VAL (1 << 0) + +#define LCEC_EL5101_CTRL_EN_LATCH_EXTN (1 << 3) +#define LCEC_EL5101_CTRL_CNT_SET (1 << 2) +#define LCEC_EL5101_CTRL_EN_LATCH_EXTP (1 << 1) +#define LCEC_EL5101_CTRL_EN_LATC (1 << 0) + +typedef struct { + hal_bit_t *ena_latch_c; + hal_bit_t *ena_latch_ext_pos; + hal_bit_t *ena_latch_ext_neg; + hal_bit_t *reset; + hal_bit_t *inext; + hal_bit_t *overflow; + hal_bit_t *underflow; + hal_bit_t *latch_c_valid; + hal_bit_t *latch_ext_valid; + hal_bit_t *set_raw_count; + hal_s32_t *set_raw_count_val; + hal_s32_t *raw_count; + hal_s32_t *raw_latch; + hal_u32_t *raw_frequency; + hal_u32_t *raw_period; + hal_u32_t *raw_window; + hal_s32_t *count; + hal_float_t *pos_scale; + hal_float_t *pos; + hal_float_t *period; + hal_float_t *frequency; + + unsigned int status_pdo_os; + unsigned int value_pdo_os; + unsigned int latch_pdo_os; + unsigned int frequency_pdo_os; + unsigned int period_pdo_os; + unsigned int window_pdo_os; + unsigned int control_pdo_os; + unsigned int setval_pdo_os; + + int do_init; + int16_t last_count; + double old_scale; + double scale; + + int last_operational; +} lcec_el5101_data_t; + +static ec_pdo_entry_info_t lcec_el5101_in[] = { + {0x6000, 0x01, 8}, // Status + {0x6000, 0x02, 16}, // Value + {0x6000, 0x03, 16} // Latch +}; + +static ec_pdo_entry_info_t lcec_el5101_period[] = { + {0x6000, 0x04, 32}, // Frequency + {0x6000, 0x05, 16}, // Period + {0x6000, 0x06, 16} // Window +}; + +static ec_pdo_entry_info_t lcec_el5101_out[] = { + {0x7000, 0x01, 8}, // Ctrl + {0x7000, 0x02, 16} // Value +}; + +static ec_pdo_info_t lcec_el5101_pdos_out[] = { + {0x1600, 2, lcec_el5101_out} +}; + +static ec_pdo_info_t lcec_el5101_pdos_in[] = { + {0x1A00, 3, lcec_el5101_in}, + {0x1A02, 3, lcec_el5101_period} +}; + +static ec_sync_info_t lcec_el5101_syncs[] = { + {0, EC_DIR_OUTPUT, 0, NULL}, + {1, EC_DIR_INPUT, 0, NULL}, + {2, EC_DIR_OUTPUT, 1, lcec_el5101_pdos_out}, + {3, EC_DIR_INPUT, 2, lcec_el5101_pdos_in}, + {0xff} +}; + + +void lcec_el5101_read(struct lcec_slave *slave, long period); +void lcec_el5101_write(struct lcec_slave *slave, long period); + +int lcec_el5101_init(int comp_id, struct lcec_slave *slave, ec_pdo_entry_reg_t *pdo_entry_regs) { + lcec_master_t *master = slave->master; + lcec_el5101_data_t *hal_data; + int err; + + // initialize callbacks + slave->proc_read = lcec_el5101_read; + slave->proc_write = lcec_el5101_write; + + // alloc hal memory + if ((hal_data = hal_malloc(sizeof(lcec_el5101_data_t))) == NULL) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "hal_malloc() for slave %s.%s failed\n", master->name, slave->name); + return -EIO; + } + memset(hal_data, 0, sizeof(lcec_el5101_data_t)); + slave->hal_data = hal_data; + + // initializer sync info + slave->sync_info = lcec_el5101_syncs; + + // initialize global data + hal_data->last_operational = 0; + + // initialize POD entries + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6000, 0x01, &hal_data->status_pdo_os, NULL); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6000, 0x02, &hal_data->value_pdo_os, NULL); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6000, 0x03, &hal_data->latch_pdo_os, NULL); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6000, 0x04, &hal_data->frequency_pdo_os, NULL); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6000, 0x05, &hal_data->period_pdo_os, NULL); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6000, 0x06, &hal_data->window_pdo_os, NULL); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x7000, 0x01, &hal_data->control_pdo_os, NULL); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x7000, 0x02, &hal_data->setval_pdo_os, NULL); + + // export pins + if ((err = hal_pin_bit_newf(HAL_IO, &(hal_data->ena_latch_c), comp_id, "%s.%s.%s.enc-index-c-enable", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-index-c-enable failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_IO, &(hal_data->ena_latch_ext_pos), comp_id, "%s.%s.%s.enc-index-ext-pos-enable", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-index-ext-pos-enable failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_IO, &(hal_data->ena_latch_ext_neg), comp_id, "%s.%s.%s.enc-index-ext-neg-enable", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-index-ext-neg-enable failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_IN, &(hal_data->reset), comp_id, "%s.%s.%s.enc-reset", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-reset failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(hal_data->inext), comp_id, "%s.%s.%s.enc-inext", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-inext failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(hal_data->overflow), comp_id, "%s.%s.%s.enc-overflow", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-overflow failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(hal_data->underflow), comp_id, "%s.%s.%s.enc-underflow", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-underflow failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(hal_data->latch_c_valid), comp_id, "%s.%s.%s.enc-latch-c-valid", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-latch-c-valid failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(hal_data->latch_ext_valid), comp_id, "%s.%s.%s.enc-latch-ext-valid", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-latch-ext-valid failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_IO, &(hal_data->set_raw_count), comp_id, "%s.%s.%s.enc-set-raw-count", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-set-raw-count failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_s32_newf(HAL_IN, &(hal_data->set_raw_count_val), comp_id, "%s.%s.%s.enc-set-raw-count-val", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-set-raw-count-val failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_s32_newf(HAL_OUT, &(hal_data->raw_count), comp_id, "%s.%s.%s.enc-raw-count", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-raw-count failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_s32_newf(HAL_OUT, &(hal_data->count), comp_id, "%s.%s.%s.enc-count", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-count failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_s32_newf(HAL_OUT, &(hal_data->raw_latch), comp_id, "%s.%s.%s.enc-raw-latch", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-raw-latch failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_u32_newf(HAL_OUT, &(hal_data->raw_frequency), comp_id, "%s.%s.%s.enc-raw-freq", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-raw-freq failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_u32_newf(HAL_OUT, &(hal_data->raw_period), comp_id, "%s.%s.%s.enc-raw-period", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-raw-period failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_u32_newf(HAL_OUT, &(hal_data->raw_window), comp_id, "%s.%s.%s.enc-raw-window", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-raw-window failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_float_newf(HAL_OUT, &(hal_data->pos), comp_id, "%s.%s.%s.enc-pos", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-pos failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_float_newf(HAL_OUT, &(hal_data->period), comp_id, "%s.%s.%s.enc-period", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-period failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_float_newf(HAL_OUT, &(hal_data->frequency), comp_id, "%s.%s.%s.enc-frequency", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-frequency failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_float_newf(HAL_IO, &(hal_data->pos_scale), comp_id, "%s.%s.%s.enc-pos-scale", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-scale failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + + // initialize pins + *(hal_data->ena_latch_c) = 0; + *(hal_data->ena_latch_ext_pos) = 0; + *(hal_data->ena_latch_ext_neg) = 0; + *(hal_data->reset) = 0; + *(hal_data->inext) = 0; + *(hal_data->overflow) = 0; + *(hal_data->underflow) = 0; + *(hal_data->latch_c_valid) = 0; + *(hal_data->latch_ext_valid) = 0; + *(hal_data->set_raw_count) = 0; + *(hal_data->set_raw_count_val) = 0; + *(hal_data->raw_count) = 0; + *(hal_data->raw_latch) = 0; + *(hal_data->raw_frequency) = 0; + *(hal_data->raw_period) = 0; + *(hal_data->raw_window) = 0; + *(hal_data->count) = 0; + *(hal_data->pos) = 0; + *(hal_data->period) = 0; + *(hal_data->frequency) = 0; + *(hal_data->pos_scale) = 1.0; + + // initialize variables + hal_data->do_init = 1; + hal_data->last_count = 0; + hal_data->old_scale = *(hal_data->pos_scale) + 1.0; + hal_data->scale = 1.0; + + return 0; +} + +void lcec_el5101_read(struct lcec_slave *slave, long period) { + lcec_master_t *master = slave->master; + lcec_el5101_data_t *hal_data = (lcec_el5101_data_t *) slave->hal_data; + uint8_t *pd = master->process_data; + uint8_t raw_status; + int16_t raw_count, raw_latch, raw_delta; + uint16_t raw_period, raw_window; + uint32_t raw_frequency; + + // wait for slave to be operational + if (!slave->state.operational) { + hal_data->last_operational = 0; + return; + } + + // check for change in scale value + if (*(hal_data->pos_scale) != hal_data->old_scale) { + // scale value has changed, test and update it + if ((*(hal_data->pos_scale) < 1e-20) && (*(hal_data->pos_scale) > -1e-20)) { + // value too small, divide by zero is a bad thing + *(hal_data->pos_scale) = 1.0; + } + // save new scale to detect future changes + hal_data->old_scale = *(hal_data->pos_scale); + // we actually want the reciprocal + hal_data->scale = 1.0 / *(hal_data->pos_scale); + } + + // get bit states + raw_status = EC_READ_U8(&pd[hal_data->status_pdo_os]); + *(hal_data->inext) = raw_status & LCEC_EL5101_STATUS_INPUT; + *(hal_data->overflow) = raw_status & LCEC_EL5101_STATUS_OVERFLOW; + *(hal_data->underflow) = raw_status & LCEC_EL5101_STATUS_UNDERFLOW; + *(hal_data->latch_ext_valid) = raw_status & LCEC_EL5101_STATUS_LAT_EXT_VAL; + *(hal_data->latch_c_valid) = raw_status & LCEC_EL5101_STATUS_LATC_VAL; + + // read raw values + raw_count = EC_READ_S16(&pd[hal_data->value_pdo_os]); + raw_latch = EC_READ_S16(&pd[hal_data->latch_pdo_os]); + raw_frequency = EC_READ_U32(&pd[hal_data->frequency_pdo_os]); + raw_period = EC_READ_U16(&pd[hal_data->period_pdo_os]); + raw_window = EC_READ_U16(&pd[hal_data->window_pdo_os]); + + // check for operational change of slave + if (!hal_data->last_operational) { + hal_data->last_count = raw_count; + } + + // check for counter set done + if (raw_status & LCEC_EL5101_STATUS_CNTSET_ACC) { + hal_data->last_count = raw_count; + *(hal_data->set_raw_count) = 0; + } + + // update raw values + if (! *(hal_data->set_raw_count)) { + *(hal_data->raw_count) = raw_count; + *(hal_data->raw_frequency) = raw_frequency; + *(hal_data->raw_period) = raw_period; + *(hal_data->raw_window) = raw_window; + } + + // handle initialization + if (hal_data->do_init || *(hal_data->reset)) { + hal_data->do_init = 0; + hal_data->last_count = raw_count; + *(hal_data->count) = 0; + } + + // handle index + if (*(hal_data->latch_ext_valid)) { + *(hal_data->raw_latch) = raw_latch; + hal_data->last_count = raw_latch; + *(hal_data->count) = 0; + *(hal_data->ena_latch_ext_pos) = 0; + *(hal_data->ena_latch_ext_neg) = 0; + } + if (*(hal_data->latch_c_valid)) { + *(hal_data->raw_latch) = raw_latch; + hal_data->last_count = raw_latch; + *(hal_data->count) = 0; + *(hal_data->ena_latch_c) = 0; + } + + // compute net counts + raw_delta = raw_count - hal_data->last_count; + hal_data->last_count = raw_count; + *(hal_data->count) += raw_delta; + + // scale count to make floating point position + *(hal_data->pos) = *(hal_data->count) * hal_data->scale; + + // scale period + *(hal_data->frequency) = ((double) (*(hal_data->raw_frequency))) * LCEC_EL5101_FREQUENCY_SCALE; + *(hal_data->period) = ((double) (*(hal_data->raw_period))) * LCEC_EL5101_PERIOD_SCALE; + + hal_data->last_operational = 1; +} + +void lcec_el5101_write(struct lcec_slave *slave, long period) { + lcec_master_t *master = slave->master; + lcec_el5101_data_t *hal_data = (lcec_el5101_data_t *) slave->hal_data; + uint8_t *pd = master->process_data; + uint8_t raw_ctrl; + + // build control byte + raw_ctrl = 0; + if (*(hal_data->ena_latch_ext_neg)) { + raw_ctrl |= LCEC_EL5101_CTRL_EN_LATCH_EXTN; + } + if (*(hal_data->set_raw_count)) { + raw_ctrl |= LCEC_EL5101_CTRL_CNT_SET; + } + if (*(hal_data->ena_latch_ext_pos)) { + raw_ctrl |= LCEC_EL5101_CTRL_EN_LATCH_EXTP; + } + if (*(hal_data->ena_latch_c)) { + raw_ctrl |= LCEC_EL5101_CTRL_EN_LATC; + } + + // set output data + EC_WRITE_U8(&pd[hal_data->control_pdo_os], raw_ctrl); + EC_WRITE_S16(&pd[hal_data->setval_pdo_os], *(hal_data->set_raw_count_val)); +} + diff --git a/src/lcec_el5101.h b/src/lcec_el5101.h new file mode 100644 index 00000000..5bd3dde0 --- /dev/null +++ b/src/lcec_el5101.h @@ -0,0 +1,34 @@ +// +// Copyright (C) 2011 Sascha Ittner +// +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2 of the License, or +// (at your option) any later version. +// +// This program 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 this program; if not, write to the Free Software +// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA +// +#ifndef _LCEC_EL5101_H_ +#define _LCEC_EL5101_H_ + +#include "lcec.h" + +#define LCEC_EL5101_VID LCEC_BECKHOFF_VID +#define LCEC_EL5101_PID 0x13ed3052 + +#define LCEC_EL5101_PDOS 8 + +#define LCEC_EL5101_PERIOD_SCALE 500e-9 +#define LCEC_EL5101_FREQUENCY_SCALE 5e-2 + +int lcec_el5101_init(int comp_id, struct lcec_slave *slave, ec_pdo_entry_reg_t *pdo_entry_regs); + +#endif + diff --git a/src/lcec_el5151.c b/src/lcec_el5151.c new file mode 100644 index 00000000..fbc0ba2f --- /dev/null +++ b/src/lcec_el5151.c @@ -0,0 +1,413 @@ +// +// Copyright (C) 2012 Sascha Ittner +// +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2 of the License, or +// (at your option) any later version. +// +// This program 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 this program; if not, write to the Free Software +// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA +// + +#include "lcec.h" +#include "lcec_el5151.h" + +typedef struct { + hal_bit_t *ena_latch_c; + hal_bit_t *ena_latch_ext_pos; + hal_bit_t *ena_latch_ext_neg; + hal_bit_t *reset; + hal_bit_t *ina; + hal_bit_t *inb; + hal_bit_t *inc; + hal_bit_t *inext; + hal_bit_t *expol_stall; + hal_bit_t *sync_err; + hal_bit_t *latch_c_valid; + hal_bit_t *latch_ext_valid; + hal_bit_t *tx_toggle; + hal_bit_t *set_raw_count; + hal_s32_t *set_raw_count_val; + hal_s32_t *raw_count; + hal_s32_t *raw_latch; + hal_u32_t *raw_period; + hal_s32_t *count; + hal_float_t *pos_scale; + hal_float_t *pos; + hal_float_t *period; + + unsigned int ena_latch_c_pdo_os; + unsigned int ena_latch_c_pdo_bp; + unsigned int ena_latch_ext_pos_pdo_os; + unsigned int ena_latch_ext_pos_pdo_bp; + unsigned int ena_latch_ext_neg_pdo_os; + unsigned int ena_latch_ext_neg_pdo_bp; + unsigned int set_count_pdo_os; + unsigned int set_count_pdo_bp; + unsigned int set_count_val_pdo_os; + unsigned int set_count_done_pdo_os; + unsigned int set_count_done_pdo_bp; + unsigned int latch_c_valid_pdo_os; + unsigned int latch_c_valid_pdo_bp; + unsigned int latch_ext_valid_pdo_os; + unsigned int latch_ext_valid_pdo_bp; + unsigned int expol_stall_pdo_os; + unsigned int expol_stall_pdo_bp; + unsigned int ina_pdo_os; + unsigned int ina_pdo_bp; + unsigned int inb_pdo_os; + unsigned int inb_pdo_bp; + unsigned int inc_pdo_os; + unsigned int inc_pdo_bp; + unsigned int inext_pdo_os; + unsigned int inext_pdo_bp; + unsigned int sync_err_pdo_os; + unsigned int sync_err_pdo_bp; + unsigned int tx_toggle_pdo_os; + unsigned int tx_toggle_pdo_bp; + unsigned int count_pdo_os; + unsigned int latch_pdo_os; + unsigned int period_pdo_os; + + int do_init; + int32_t last_count; + double old_scale; + double scale; + + int last_operational; +} lcec_el5151_data_t; + +static ec_pdo_entry_info_t lcec_el5151_in[] = { + {0x6000, 0x01, 1}, // Latch C valid + {0x6000, 0x02, 1}, // Latch extern valid + {0x6000, 0x03, 1}, // Set counter done + {0x0000, 0x00, 4}, // Gap + {0x6000, 0x08, 1}, // Extrapolation stall + {0x6000, 0x09, 1}, // Status of input A + {0x6000, 0x0a, 1}, // Status of input B + {0x6000, 0x0b, 1}, // Status of input C + {0x0000, 0x00, 1}, // Gap + {0x6000, 0x0d, 1}, // Status of extern latch + {0x1c32, 0x20, 1}, // Sync error + {0x0000, 0x00, 1}, // Gap + {0x1800, 0x09, 1}, // TxPDO Toggle + {0x6000, 0x11, 32}, // Counter value + {0x6000, 0x12, 32} // Latch value +}; + +static ec_pdo_entry_info_t lcec_el5151_period[] = { + {0x6000, 0x14, 32} // Period value +}; + +static ec_pdo_entry_info_t lcec_el5151_out[] = { + {0x7000, 0x01, 1}, // Enable latch C + {0x7000, 0x02, 1}, // Enable latch extern on positive edge + {0x7000, 0x03, 1}, // Set counter + {0x7000, 0x04, 1}, // Enable latch extern on negative edge + {0x0000, 0x00, 4}, // Gap + {0x0000, 0x00, 8}, // Gap + {0x7000, 0x11, 32} // Set counter value +}; + +static ec_pdo_info_t lcec_el5151_pdos_out[] = { + {0x1600, 7, lcec_el5151_out} +}; + +static ec_pdo_info_t lcec_el5151_pdos_in[] = { + {0x1A00, 15, lcec_el5151_in}, + {0x1A02, 1, lcec_el5151_period} +}; + +static ec_sync_info_t lcec_el5151_syncs[] = { + {0, EC_DIR_OUTPUT, 0, NULL}, + {1, EC_DIR_INPUT, 0, NULL}, + {2, EC_DIR_OUTPUT, 1, lcec_el5151_pdos_out}, + {3, EC_DIR_INPUT, 2, lcec_el5151_pdos_in}, + {0xff} +}; + + +void lcec_el5151_read(struct lcec_slave *slave, long period); +void lcec_el5151_write(struct lcec_slave *slave, long period); + +int lcec_el5151_init(int comp_id, struct lcec_slave *slave, ec_pdo_entry_reg_t *pdo_entry_regs) { + lcec_master_t *master = slave->master; + lcec_el5151_data_t *hal_data; + int err; + + // initialize callbacks + slave->proc_read = lcec_el5151_read; + slave->proc_write = lcec_el5151_write; + + // alloc hal memory + if ((hal_data = hal_malloc(sizeof(lcec_el5151_data_t))) == NULL) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "hal_malloc() for slave %s.%s failed\n", master->name, slave->name); + return -EIO; + } + memset(hal_data, 0, sizeof(lcec_el5151_data_t)); + slave->hal_data = hal_data; + + // initializer sync info + slave->sync_info = lcec_el5151_syncs; + + // initialize global data + hal_data->last_operational = 0; + + // initialize POD entries + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6000, 0x01, &hal_data->latch_c_valid_pdo_os, &hal_data->latch_c_valid_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6000, 0x02, &hal_data->latch_ext_valid_pdo_os, &hal_data->latch_ext_valid_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6000, 0x03, &hal_data->set_count_done_pdo_os, &hal_data->set_count_done_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6000, 0x08, &hal_data->expol_stall_pdo_os, &hal_data->expol_stall_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6000, 0x09, &hal_data->ina_pdo_os, &hal_data->ina_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6000, 0x0a, &hal_data->inb_pdo_os, &hal_data->inb_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6000, 0x0b, &hal_data->inc_pdo_os, &hal_data->inc_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6000, 0x0d, &hal_data->inext_pdo_os, &hal_data->inext_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x1c32, 0x20, &hal_data->sync_err_pdo_os, &hal_data->sync_err_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x1800, 0x09, &hal_data->tx_toggle_pdo_os, &hal_data->tx_toggle_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6000, 0x11, &hal_data->count_pdo_os, NULL); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6000, 0x12, &hal_data->latch_pdo_os, NULL); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6000, 0x14, &hal_data->period_pdo_os, NULL); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x7000, 0x01, &hal_data->ena_latch_c_pdo_os, &hal_data->ena_latch_c_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x7000, 0x02, &hal_data->ena_latch_ext_pos_pdo_os, &hal_data->ena_latch_ext_pos_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x7000, 0x03, &hal_data->set_count_pdo_os, &hal_data->set_count_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x7000, 0x04, &hal_data->ena_latch_ext_neg_pdo_os, &hal_data->ena_latch_ext_neg_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x7000, 0x11, &hal_data->set_count_val_pdo_os, NULL); + + // export pins + if ((err = hal_pin_bit_newf(HAL_IO, &(hal_data->ena_latch_c), comp_id, "%s.%s.%s.enc-index-c-enable", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-index-c-enable failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_IO, &(hal_data->ena_latch_ext_pos), comp_id, "%s.%s.%s.enc-index-ext-pos-enable", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-index-ext-pos-enable failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_IO, &(hal_data->ena_latch_ext_neg), comp_id, "%s.%s.%s.enc-index-ext-neg-enable", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-index-ext-neg-enable failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_IN, &(hal_data->reset), comp_id, "%s.%s.%s.enc-reset", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-reset failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(hal_data->ina), comp_id, "%s.%s.%s.enc-ina", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-ina failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(hal_data->inb), comp_id, "%s.%s.%s.enc-inb", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-inb failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(hal_data->inc), comp_id, "%s.%s.%s.enc-inc", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-inc failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(hal_data->inext), comp_id, "%s.%s.%s.enc-inext", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-inext failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(hal_data->expol_stall), comp_id, "%s.%s.%s.enc-expol-stall", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-expol-stall failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(hal_data->sync_err), comp_id, "%s.%s.%s.enc-sync-error", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-sync-error failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(hal_data->latch_c_valid), comp_id, "%s.%s.%s.enc-latch-c-valid", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-latch-c-valid failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(hal_data->latch_ext_valid), comp_id, "%s.%s.%s.enc-latch-ext-valid", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-latch-ext-valid failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(hal_data->tx_toggle), comp_id, "%s.%s.%s.enc-tx-toggle", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-tx-toggle failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_IO, &(hal_data->set_raw_count), comp_id, "%s.%s.%s.enc-set-raw-count", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-set-raw-count failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_s32_newf(HAL_IN, &(hal_data->set_raw_count_val), comp_id, "%s.%s.%s.enc-set-raw-count-val", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-set-raw-count-val failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_s32_newf(HAL_OUT, &(hal_data->raw_count), comp_id, "%s.%s.%s.enc-raw-count", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-raw-count failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_s32_newf(HAL_OUT, &(hal_data->count), comp_id, "%s.%s.%s.enc-count", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-count failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_s32_newf(HAL_OUT, &(hal_data->raw_latch), comp_id, "%s.%s.%s.enc-raw-latch", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-raw-latch failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_u32_newf(HAL_OUT, &(hal_data->raw_period), comp_id, "%s.%s.%s.enc-raw-period", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-raw-period failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_float_newf(HAL_OUT, &(hal_data->pos), comp_id, "%s.%s.%s.enc-pos", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-pos failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_float_newf(HAL_OUT, &(hal_data->period), comp_id, "%s.%s.%s.enc-period", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-period failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_float_newf(HAL_IO, &(hal_data->pos_scale), comp_id, "%s.%s.%s.enc-pos-scale", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-scale failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + + // initialize pins + *(hal_data->ena_latch_c) = 0; + *(hal_data->ena_latch_ext_pos) = 0; + *(hal_data->ena_latch_ext_neg) = 0; + *(hal_data->reset) = 0; + *(hal_data->ina) = 0; + *(hal_data->inb) = 0; + *(hal_data->inc) = 0; + *(hal_data->inext) = 0; + *(hal_data->expol_stall) = 0; + *(hal_data->sync_err) = 0; + *(hal_data->latch_c_valid) = 0; + *(hal_data->latch_ext_valid) = 0; + *(hal_data->tx_toggle) = 0; + *(hal_data->set_raw_count) = 0; + *(hal_data->set_raw_count_val) = 0; + *(hal_data->raw_count) = 0; + *(hal_data->raw_latch) = 0; + *(hal_data->raw_period) = 0; + *(hal_data->count) = 0; + *(hal_data->pos) = 0; + *(hal_data->period) = 0; + *(hal_data->pos_scale) = 1.0; + + // initialize variables + hal_data->do_init = 1; + hal_data->last_count = 0; + hal_data->old_scale = *(hal_data->pos_scale) + 1.0; + hal_data->scale = 1.0; + + return 0; +} + +void lcec_el5151_read(struct lcec_slave *slave, long period) { + lcec_master_t *master = slave->master; + lcec_el5151_data_t *hal_data = (lcec_el5151_data_t *) slave->hal_data; + uint8_t *pd = master->process_data; + int32_t raw_count, raw_latch, raw_delta; + uint32_t raw_period; + + // wait for slave to be operational + if (!slave->state.operational) { + hal_data->last_operational = 0; + return; + } + + // check for change in scale value + if (*(hal_data->pos_scale) != hal_data->old_scale) { + // scale value has changed, test and update it + if ((*(hal_data->pos_scale) < 1e-20) && (*(hal_data->pos_scale) > -1e-20)) { + // value too small, divide by zero is a bad thing + *(hal_data->pos_scale) = 1.0; + } + // save new scale to detect future changes + hal_data->old_scale = *(hal_data->pos_scale); + // we actually want the reciprocal + hal_data->scale = 1.0 / *(hal_data->pos_scale); + } + + // get bit states + *(hal_data->ina) = EC_READ_BIT(&pd[hal_data->ina_pdo_os], hal_data->ina_pdo_bp); + *(hal_data->inb) = EC_READ_BIT(&pd[hal_data->inb_pdo_os], hal_data->inb_pdo_bp); + *(hal_data->inc) = EC_READ_BIT(&pd[hal_data->inc_pdo_os], hal_data->inc_pdo_bp); + *(hal_data->inext) = EC_READ_BIT(&pd[hal_data->inext_pdo_os], hal_data->inext_pdo_bp); + *(hal_data->expol_stall) = EC_READ_BIT(&pd[hal_data->expol_stall_pdo_os], hal_data->expol_stall_pdo_bp); + *(hal_data->sync_err) = EC_READ_BIT(&pd[hal_data->sync_err_pdo_os], hal_data->sync_err_pdo_bp); + *(hal_data->latch_c_valid) = EC_READ_BIT(&pd[hal_data->latch_c_valid_pdo_os], hal_data->latch_c_valid_pdo_bp); + *(hal_data->latch_ext_valid) = EC_READ_BIT(&pd[hal_data->latch_ext_valid_pdo_os], hal_data->latch_ext_valid_pdo_bp); + *(hal_data->tx_toggle) = EC_READ_BIT(&pd[hal_data->tx_toggle_pdo_os], hal_data->tx_toggle_pdo_bp); + + // read raw values + raw_count = EC_READ_S32(&pd[hal_data->count_pdo_os]); + raw_latch = EC_READ_S32(&pd[hal_data->latch_pdo_os]); + raw_period = EC_READ_U32(&pd[hal_data->period_pdo_os]); + + // check for operational change of slave + if (!hal_data->last_operational) { + hal_data->last_count = raw_count; + } + + // check for counter set done + if (EC_READ_BIT(&pd[hal_data->set_count_done_pdo_os], hal_data->set_count_done_pdo_bp)) { + hal_data->last_count = raw_count; + *(hal_data->set_raw_count) = 0; + } + + // update raw values + if (! *(hal_data->set_raw_count)) { + *(hal_data->raw_count) = raw_count; + *(hal_data->raw_period) = raw_period; + } + + // handle initialization + if (hal_data->do_init || *(hal_data->reset)) { + hal_data->do_init = 0; + hal_data->last_count = raw_count; + *(hal_data->count) = 0; + } + + // handle index + if (*(hal_data->latch_ext_valid)) { + *(hal_data->raw_latch) = raw_latch; + hal_data->last_count = raw_latch; + *(hal_data->count) = 0; + *(hal_data->ena_latch_ext_pos) = 0; + *(hal_data->ena_latch_ext_neg) = 0; + } + if (*(hal_data->latch_c_valid)) { + *(hal_data->raw_latch) = raw_latch; + hal_data->last_count = raw_latch; + *(hal_data->count) = 0; + *(hal_data->ena_latch_c) = 0; + } + + // compute net counts + raw_delta = raw_count - hal_data->last_count; + hal_data->last_count = raw_count; + *(hal_data->count) += raw_delta; + + // scale count to make floating point position + *(hal_data->pos) = *(hal_data->count) * hal_data->scale; + + // scale period + *(hal_data->period) = ((double) (*(hal_data->raw_period))) * LCEC_EL5151_PERIOD_SCALE; + + hal_data->last_operational = 1; +} + +void lcec_el5151_write(struct lcec_slave *slave, long period) { + lcec_master_t *master = slave->master; + lcec_el5151_data_t *hal_data = (lcec_el5151_data_t *) slave->hal_data; + uint8_t *pd = master->process_data; + + // set output data + EC_WRITE_BIT(&pd[hal_data->set_count_pdo_os], hal_data->set_count_pdo_bp, *(hal_data->set_raw_count)); + EC_WRITE_BIT(&pd[hal_data->ena_latch_c_pdo_os], hal_data->ena_latch_c_pdo_bp, *(hal_data->ena_latch_c)); + EC_WRITE_BIT(&pd[hal_data->ena_latch_ext_pos_pdo_os], hal_data->ena_latch_ext_pos_pdo_bp, *(hal_data->ena_latch_ext_pos)); + EC_WRITE_BIT(&pd[hal_data->ena_latch_ext_neg_pdo_os], hal_data->ena_latch_ext_neg_pdo_bp, *(hal_data->ena_latch_ext_neg)); + EC_WRITE_S32(&pd[hal_data->set_count_val_pdo_os], *(hal_data->set_raw_count_val)); +} + diff --git a/src/lcec_el5151.h b/src/lcec_el5151.h new file mode 100644 index 00000000..9ca60962 --- /dev/null +++ b/src/lcec_el5151.h @@ -0,0 +1,33 @@ +// +// Copyright (C) 2011 Sascha Ittner +// +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2 of the License, or +// (at your option) any later version. +// +// This program 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 this program; if not, write to the Free Software +// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA +// +#ifndef _LCEC_EL5151_H_ +#define _LCEC_EL5151_H_ + +#include "lcec.h" + +#define LCEC_EL5151_VID LCEC_BECKHOFF_VID +#define LCEC_EL5151_PID 0x141f3052 + +#define LCEC_EL5151_PDOS 18 + +#define LCEC_EL5151_PERIOD_SCALE 1e-7 + +int lcec_el5151_init(int comp_id, struct lcec_slave *slave, ec_pdo_entry_reg_t *pdo_entry_regs); + +#endif + diff --git a/src/lcec_el5152.c b/src/lcec_el5152.c new file mode 100644 index 00000000..b3c25c57 --- /dev/null +++ b/src/lcec_el5152.c @@ -0,0 +1,391 @@ +// +// Copyright (C) 2011 Sascha Ittner +// +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2 of the License, or +// (at your option) any later version. +// +// This program 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 this program; if not, write to the Free Software +// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA +// + +#include "lcec.h" +#include "lcec_el5152.h" + +typedef struct { + hal_bit_t *index; + hal_bit_t *index_ena; + hal_bit_t *reset; + hal_bit_t *ina; + hal_bit_t *inb; + hal_bit_t *expol_stall; + hal_bit_t *tx_toggle; + hal_bit_t *set_raw_count; + hal_s32_t *set_raw_count_val; + hal_s32_t *raw_count; + hal_u32_t *raw_period; + hal_s32_t *count; + hal_float_t *pos_scale; + hal_float_t *pos; + hal_float_t *period; + + unsigned int set_count_pdo_os; + unsigned int set_count_pdo_bp; + unsigned int set_count_val_pdo_os; + unsigned int set_count_done_pdo_os; + unsigned int set_count_done_pdo_bp; + unsigned int expol_stall_pdo_os; + unsigned int expol_stall_pdo_bp; + unsigned int ina_pdo_os; + unsigned int ina_pdo_bp; + unsigned int inb_pdo_os; + unsigned int inb_pdo_bp; + unsigned int tx_toggle_pdo_os; + unsigned int tx_toggle_pdo_bp; + unsigned int count_pdo_os; + unsigned int period_pdo_os; + + int do_init; + int32_t last_count; + int last_index; + double old_scale; + double scale; +} lcec_el5152_chan_t; + +typedef struct { + lcec_el5152_chan_t chans[LCEC_EL5152_CHANS]; + int last_operational; +} lcec_el5152_data_t; + +static ec_pdo_entry_info_t lcec_el5152_channel1_in[] = { + {0x0000, 0x00, 2}, // Gap + {0x6000, 0x03, 1}, // Set counter done + {0x0000, 0x00, 4}, // Gap + {0x6000, 0x08, 1}, // Extrapolation stall + {0x6000, 0x09, 1}, // State input A + {0x6000, 0x0a, 1}, // State input B + {0x0000, 0x00, 3}, // Gap + {0x1c32, 0x20, 1}, // Sync error + {0x0000, 0x00, 1}, // Gap + {0x1800, 0x09, 1}, // TxPDO toggle + {0x6000, 0x11, 32} // Counter value +}; + +static ec_pdo_entry_info_t lcec_el5152_channel2_in[] = { + {0x0000, 0x00, 2}, // Gap + {0x6010, 0x03, 1}, // Set counter done + {0x0000, 0x00, 4}, // Gap + {0x6010, 0x08, 1}, // Extrapolation stall + {0x6010, 0x09, 1}, // State input A + {0x6010, 0x0a, 1}, // State input B + {0x0000, 0x00, 3}, // Gap + {0x1c32, 0x20, 1}, // Sync error + {0x0000, 0x00, 1}, // Gap + {0x1804, 0x09, 1}, // TxPDO toggle + {0x6010, 0x11, 32} // Counter value +}; + +static ec_pdo_entry_info_t lcec_el5152_channel1_period[] = { + {0x6000, 0x14, 32} // Period value +}; + +static ec_pdo_entry_info_t lcec_el5152_channel2_period[] = { + {0x6010, 0x14, 32} // Period value +}; + +static ec_pdo_entry_info_t lcec_el5152_channel1_out[] = { + {0x0000, 0x00, 1}, // Gap + {0x0000, 0x00, 1}, // Gap + {0x7000, 0x03, 1}, // Set counter + {0x0000, 0x00, 1}, // Gap + {0x0000, 0x00, 4}, // Gap + {0x0000, 0x00, 8}, // Gap + {0x7000, 0x11, 32} // Set counter value +}; + +static ec_pdo_entry_info_t lcec_el5152_channel2_out[] = { + {0x0000, 0x00, 1}, // Gap + {0x0000, 0x00, 1}, // Gap + {0x7010, 0x03, 1}, // Set counter + {0x0000, 0x00, 1}, // Gap + {0x0000, 0x00, 4}, // Gap + {0x0000, 0x00, 8}, // Gap + {0x7010, 0x11, 32} // Set counter value +}; + +static ec_pdo_info_t lcec_el5152_pdos_out[] = { + {0x1600, 7, lcec_el5152_channel1_out}, + {0x1602, 7, lcec_el5152_channel2_out} +}; + +static ec_pdo_info_t lcec_el5152_pdos_in[] = { + {0x1A00, 11, lcec_el5152_channel1_in}, + {0x1A02, 1, lcec_el5152_channel1_period}, + {0x1A04, 11, lcec_el5152_channel2_in}, + {0x1A06, 1, lcec_el5152_channel2_period} +}; + +static ec_sync_info_t lcec_el5152_syncs[] = { + {0, EC_DIR_OUTPUT, 0, NULL}, + {1, EC_DIR_INPUT, 0, NULL}, + {2, EC_DIR_OUTPUT, 2, lcec_el5152_pdos_out}, + {3, EC_DIR_INPUT, 4, lcec_el5152_pdos_in}, + {0xff} +}; + + +void lcec_el5152_read(struct lcec_slave *slave, long period); +void lcec_el5152_write(struct lcec_slave *slave, long period); + +int lcec_el5152_init(int comp_id, struct lcec_slave *slave, ec_pdo_entry_reg_t *pdo_entry_regs) { + lcec_master_t *master = slave->master; + lcec_el5152_data_t *hal_data; + int i; + lcec_el5152_chan_t *chan; + int err; + + // initialize callbacks + slave->proc_read = lcec_el5152_read; + slave->proc_write = lcec_el5152_write; + + // alloc hal memory + if ((hal_data = hal_malloc(sizeof(lcec_el5152_data_t))) == NULL) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "hal_malloc() for slave %s.%s failed\n", master->name, slave->name); + return -EIO; + } + memset(hal_data, 0, sizeof(lcec_el5152_data_t)); + slave->hal_data = hal_data; + + // initializer sync info + slave->sync_info = lcec_el5152_syncs; + + // initialize global data + hal_data->last_operational = 0; + + // initialize pins + for (i=0; ichans[i]; + + // initialize POD entries + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6000 + (i << 4), 0x03, &chan->set_count_done_pdo_os, &chan->set_count_done_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6000 + (i << 4), 0x08, &chan->expol_stall_pdo_os, &chan->expol_stall_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6000 + (i << 4), 0x09, &chan->ina_pdo_os, &chan->ina_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6000 + (i << 4), 0x0a, &chan->inb_pdo_os, &chan->inb_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x1800 + (i << 2), 0x09, &chan->tx_toggle_pdo_os, &chan->tx_toggle_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6000 + (i << 4), 0x11, &chan->count_pdo_os, NULL); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6000 + (i << 4), 0x14, &chan->period_pdo_os, NULL); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x7000 + (i << 4), 0x03, &chan->set_count_pdo_os, &chan->set_count_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x7000 + (i << 4), 0x11, &chan->set_count_val_pdo_os, NULL); + + // export pins + if ((err = hal_pin_bit_newf(HAL_IN, &(chan->index), comp_id, "%s.%s.%s.enc-%d-index", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-index failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_IO, &(chan->index_ena), comp_id, "%s.%s.%s.enc-%d-index-enable", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-index-enable failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_IN, &(chan->reset), comp_id, "%s.%s.%s.enc-%d-reset", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-reset failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(chan->ina), comp_id, "%s.%s.%s.enc-%d-ina", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-ina failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(chan->inb), comp_id, "%s.%s.%s.enc-%d-inb", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-inb failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(chan->expol_stall), comp_id, "%s.%s.%s.enc-%d-expol-stall", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-expol-stall failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(chan->tx_toggle), comp_id, "%s.%s.%s.enc-%d-tx-toggle", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-tx-toggle failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_IO, &(chan->set_raw_count), comp_id, "%s.%s.%s.enc-%d-set-raw-count", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-set-raw-count failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_s32_newf(HAL_IN, &(chan->set_raw_count_val), comp_id, "%s.%s.%s.enc-%d-set-raw-count-val", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-set-raw-count-val failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_s32_newf(HAL_OUT, &(chan->raw_count), comp_id, "%s.%s.%s.enc-%d-raw-count", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-raw-count failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_s32_newf(HAL_OUT, &(chan->count), comp_id, "%s.%s.%s.enc-%d-count", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-count failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_u32_newf(HAL_OUT, &(chan->raw_period), comp_id, "%s.%s.%s.enc-%d-raw-period", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-raw-period failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_float_newf(HAL_OUT, &(chan->pos), comp_id, "%s.%s.%s.enc-%d-pos", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-pos failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_float_newf(HAL_OUT, &(chan->period), comp_id, "%s.%s.%s.enc-%d-period", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-period failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_float_newf(HAL_IO, &(chan->pos_scale), comp_id, "%s.%s.%s.enc-%d-pos-scale", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-scale failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + + // initialize pins + *(chan->index) = 0; + *(chan->index_ena) = 0; + *(chan->reset) = 0; + *(chan->ina) = 0; + *(chan->inb) = 0; + *(chan->expol_stall) = 0; + *(chan->tx_toggle) = 0; + *(chan->set_raw_count) = 0; + *(chan->set_raw_count_val) = 0; + *(chan->raw_count) = 0; + *(chan->raw_period) = 0; + *(chan->count) = 0; + *(chan->pos) = 0; + *(chan->period) = 0; + *(chan->pos_scale) = 1.0; + + // initialize variables + chan->do_init = 1; + chan->last_count = 0; + chan->last_index = 0; + chan->old_scale = *(chan->pos_scale) + 1.0; + chan->scale = 1.0; + } + + return 0; +} + +void lcec_el5152_read(struct lcec_slave *slave, long period) { + lcec_master_t *master = slave->master; + lcec_el5152_data_t *hal_data = (lcec_el5152_data_t *) slave->hal_data; + uint8_t *pd = master->process_data; + int i, idx_flag; + lcec_el5152_chan_t *chan; + int32_t idx_count, raw_count, raw_delta; + uint32_t raw_period; + + // wait for slave to be operational + if (!slave->state.operational) { + hal_data->last_operational = 0; + return; + } + + // check inputs + for (i=0; ichans[i]; + + // check for change in scale value + if (*(chan->pos_scale) != chan->old_scale) { + // scale value has changed, test and update it + if ((*(chan->pos_scale) < 1e-20) && (*(chan->pos_scale) > -1e-20)) { + // value too small, divide by zero is a bad thing + *(chan->pos_scale) = 1.0; + } + // save new scale to detect future changes + chan->old_scale = *(chan->pos_scale); + // we actually want the reciprocal + chan->scale = 1.0 / *(chan->pos_scale); + } + + // get bit states + *(chan->ina) = EC_READ_BIT(&pd[chan->ina_pdo_os], chan->ina_pdo_bp); + *(chan->inb) = EC_READ_BIT(&pd[chan->inb_pdo_os], chan->inb_pdo_bp); + *(chan->expol_stall) = EC_READ_BIT(&pd[chan->expol_stall_pdo_os], chan->expol_stall_pdo_bp); + *(chan->tx_toggle) = EC_READ_BIT(&pd[chan->tx_toggle_pdo_os], chan->tx_toggle_pdo_bp); + + // read raw values + raw_count = EC_READ_S32(&pd[chan->count_pdo_os]); + raw_period = EC_READ_U32(&pd[chan->period_pdo_os]); + + // check for operational change of slave + if (!hal_data->last_operational) { + chan->last_count = raw_count; + } + + // check for counter set done + if (EC_READ_BIT(&pd[chan->set_count_done_pdo_os], chan->set_count_done_pdo_bp)) { + chan->last_count = raw_count; + *(chan->set_raw_count) = 0; + } + + // update raw values + if (! *(chan->set_raw_count)) { + *(chan->raw_count) = raw_count; + *(chan->raw_period) = raw_period; + } + + // check for index edge + idx_flag = 0; + idx_count = 0; + if (*(chan->index) && !chan->last_index) { + idx_count = raw_count; + idx_flag = 1; + } + chan->last_index = *(chan->index); + + // handle initialization + if (chan->do_init || *(chan->reset)) { + chan->do_init = 0; + chan->last_count = raw_count; + *(chan->count) = 0; + idx_flag = 0; + } + + // handle index + if (idx_flag && *(chan->index_ena)) { + chan->last_count = idx_count; + *(chan->count) = 0; + *(chan->index_ena) = 0; + } + + // compute net counts + raw_delta = raw_count - chan->last_count; + chan->last_count = raw_count; + *(chan->count) += raw_delta; + + // scale count to make floating point position + *(chan->pos) = *(chan->count) * chan->scale; + + // scale period + *(chan->period) = ((double) (*(chan->raw_period))) * LCEC_EL5152_PERIOD_SCALE; + } + + hal_data->last_operational = 1; +} + +void lcec_el5152_write(struct lcec_slave *slave, long period) { + lcec_master_t *master = slave->master; + lcec_el5152_data_t *hal_data = (lcec_el5152_data_t *) slave->hal_data; + uint8_t *pd = master->process_data; + int i; + lcec_el5152_chan_t *chan; + + // set outputs + for (i=0; ichans[i]; + + // set output data + EC_WRITE_BIT(&pd[chan->set_count_pdo_os], chan->set_count_pdo_bp, *(chan->set_raw_count)); + EC_WRITE_S32(&pd[chan->set_count_val_pdo_os], *(chan->set_raw_count_val)); + } +} + diff --git a/src/lcec_el5152.h b/src/lcec_el5152.h new file mode 100644 index 00000000..89e1fded --- /dev/null +++ b/src/lcec_el5152.h @@ -0,0 +1,34 @@ +// +// Copyright (C) 2011 Sascha Ittner +// +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2 of the License, or +// (at your option) any later version. +// +// This program 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 this program; if not, write to the Free Software +// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA +// +#ifndef _LCEC_EL5152_H_ +#define _LCEC_EL5152_H_ + +#include "lcec.h" + +#define LCEC_EL5152_VID LCEC_BECKHOFF_VID +#define LCEC_EL5152_PID 0x14203052 + +#define LCEC_EL5152_CHANS 2 +#define LCEC_EL5152_PDOS (9 * LCEC_EL5152_CHANS) + +#define LCEC_EL5152_PERIOD_SCALE 1e-7 + +int lcec_el5152_init(int comp_id, struct lcec_slave *slave, ec_pdo_entry_reg_t *pdo_entry_regs); + +#endif + diff --git a/src/lcec_el7342.c b/src/lcec_el7342.c new file mode 100644 index 00000000..a41d4d93 --- /dev/null +++ b/src/lcec_el7342.c @@ -0,0 +1,872 @@ +// +// Copyright (C) 2011 Sascha Ittner +// +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2 of the License, or +// (at your option) any later version. +// +// This program 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 this program; if not, write to the Free Software +// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA +// + +#include "lcec.h" +#include "lcec_el7342.h" + +#define INFO_SEL_STATUS_WORD 0 +#define INFO_SEL_MOTOR_VOLT 1 +#define INFO_SEL_MOTOR_CURR 2 +#define INFO_SEL_CURR_LIMIT 3 +#define INFO_SEL_CTRL_ERR 4 +#define INFO_SEL_DUTY_CYCLE 5 +#define INFO_SEL_MOTOR_VELO 7 +#define INFO_SEL_OVERLOAD_TIME 8 +#define INFO_SEL_INT_TEMP 101 +#define INFO_SEL_CTRL_VOLT 103 +#define INFO_SEL_SUPP_VOLT 104 +#define INFO_SEL_DCM_SWORD 150 +#define INFO_SEL_DCM_STATE 151 + +typedef struct { + hal_bit_t *reset; + hal_bit_t *ina; + hal_bit_t *inb; + hal_bit_t *inext; + hal_bit_t *sync_err; + hal_bit_t *expol_stall; + hal_bit_t *count_overflow; + hal_bit_t *count_underflow; + hal_bit_t *tx_toggle; + hal_bit_t *set_raw_count; + hal_s32_t *set_raw_count_val; + hal_bit_t *latch_ext_valid; + hal_bit_t *ena_latch_ext_pos; + hal_bit_t *ena_latch_ext_neg; + hal_s32_t *raw_count; + hal_s32_t *raw_latch; + hal_s32_t *count; + hal_float_t *pos_scale; + hal_float_t *pos; + + hal_bit_t *dcm_reset; + hal_bit_t *dcm_reduce_torque; + hal_bit_t *dcm_enable; + hal_bit_t *dcm_absmode; + hal_float_t *dcm_value; + hal_float_t *dcm_scale; + hal_float_t *dcm_offset; + hal_float_t *dcm_min_dc; + hal_float_t *dcm_max_dc; + hal_float_t *dcm_curr_dc; + hal_s32_t *dcm_raw_val; + hal_bit_t *dcm_ready_to_enable; + hal_bit_t *dcm_ready; + hal_bit_t *dcm_warning; + hal_bit_t *dcm_error; + hal_bit_t *dcm_move_pos; + hal_bit_t *dcm_move_neg; + hal_bit_t *dcm_torque_reduced; + hal_bit_t *dcm_din1; + hal_bit_t *dcm_din2; + hal_bit_t *dcm_sync_err; + hal_bit_t *dcm_tx_toggle; + hal_s32_t *dcm_raw_info1; + hal_s32_t *dcm_raw_info2; + hal_u32_t *dcm_sel_info1; + hal_u32_t *dcm_sel_info2; + hal_float_t *dcm_velo_fb; + hal_float_t *dcm_current_fb; + + unsigned int set_count_pdo_os; + unsigned int set_count_pdo_bp; + unsigned int set_count_val_pdo_os; + unsigned int set_count_done_pdo_os; + unsigned int set_count_done_pdo_bp; + unsigned int expol_stall_pdo_os; + unsigned int expol_stall_pdo_bp; + unsigned int ina_pdo_os; + unsigned int ina_pdo_bp; + unsigned int inb_pdo_os; + unsigned int inb_pdo_bp; + unsigned int inext_pdo_os; + unsigned int inext_pdo_bp; + unsigned int sync_err_pdo_os; + unsigned int sync_err_pdo_bp; + unsigned int tx_toggle_pdo_os; + unsigned int tx_toggle_pdo_bp; + unsigned int count_overflow_pdo_os; + unsigned int count_overflow_pdo_bp; + unsigned int count_underflow_pdo_os; + unsigned int count_underflow_pdo_bp; + unsigned int latch_ext_valid_pdo_os; + unsigned int latch_ext_valid_pdo_bp; + unsigned int ena_latch_ext_pos_pdo_os; + unsigned int ena_latch_ext_pos_pdo_bp; + unsigned int ena_latch_ext_neg_pdo_os; + unsigned int ena_latch_ext_neg_pdo_bp; + unsigned int count_pdo_os; + unsigned int latch_pdo_os; + + unsigned int dcm_ena_pdo_os; + unsigned int dcm_ena_pdo_bp; + unsigned int dcm_reset_pdo_os; + unsigned int dcm_reset_pdo_bp; + unsigned int dcm_reduce_torque_pdo_os; + unsigned int dcm_reduce_torque_pdo_bp; + unsigned int dcm_velo_pdo_os; + unsigned int dcm_ready_to_enable_pdo_os; + unsigned int dcm_ready_to_enable_pdo_bp; + unsigned int dcm_ready_pdo_os; + unsigned int dcm_ready_pdo_bp; + unsigned int dcm_warning_pdo_os; + unsigned int dcm_warning_pdo_bp; + unsigned int dcm_error_pdo_os; + unsigned int dcm_error_pdo_bp; + unsigned int dcm_move_pos_pdo_os; + unsigned int dcm_move_pos_pdo_bp; + unsigned int dcm_move_neg_pdo_os; + unsigned int dcm_move_neg_pdo_bp; + unsigned int dcm_torque_reduced_pdo_os; + unsigned int dcm_torque_reduced_pdo_bp; + unsigned int dcm_din1_pdo_os; + unsigned int dcm_din1_pdo_bp; + unsigned int dcm_din2_pdo_os; + unsigned int dcm_din2_pdo_bp; + unsigned int dcm_sync_err_pdo_os; + unsigned int dcm_sync_err_pdo_bp; + unsigned int dcm_tx_toggle_pdo_os; + unsigned int dcm_tx_toggle_pdo_bp; + unsigned int dcm_info1_pdo_os; + unsigned int dcm_info2_pdo_os; + + int enc_do_init; + int16_t enc_last_count; + double enc_old_scale; + double enc_scale_recip; + double dcm_old_scale; + double dcm_scale_recip; + + ec_sdo_request_t *sdo_info1_select; + ec_sdo_request_t *sdo_info2_select; + +} lcec_el7342_chan_t; + +typedef struct { + lcec_el7342_chan_t chans[LCEC_EL7342_CHANS]; + int last_operational; +} lcec_el7342_data_t; + +static ec_pdo_entry_info_t lcec_el7342_channel1_enc_out[] = { + {0x0000, 0x00, 1}, // Gap + {0x7000, 0x02, 1}, // Enable latch extern on positive edge + {0x7000, 0x03, 1}, // Set counter + {0x7000, 0x04, 1}, // Enable latch extern on negative edge + {0x0000, 0x00, 4}, // Gap + {0x0000, 0x00, 8}, // Gap + {0x7000, 0x11, 16} // Set counter value +}; + +static ec_pdo_entry_info_t lcec_el7342_channel2_enc_out[] = { + {0x0000, 0x00, 1}, // Gap + {0x7010, 0x02, 1}, // Enable latch extern on positive edge + {0x7010, 0x03, 1}, // Set counter + {0x7010, 0x04, 1}, // Enable latch extern on negative edge + {0x0000, 0x00, 4}, // Gap + {0x0000, 0x00, 8}, // Gap + {0x7010, 0x11, 16} // Set counter value +}; + +static ec_pdo_entry_info_t lcec_el7342_channel1_dcm_out[] = { + {0x7020, 0x01, 1}, // Enable + {0x7020, 0x02, 1}, // Reset + {0x7020, 0x03, 1}, // Reduce torque + {0x0000, 0x00, 5}, // Gap + {0x0000, 0x00, 8} // Gap +}; + +static ec_pdo_entry_info_t lcec_el7342_channel1_vel_out[] = { + {0x7020, 0x21, 16}, // Velocity +}; + +static ec_pdo_entry_info_t lcec_el7342_channel2_dcm_out[] = { + {0x7030, 0x01, 1}, // Enable + {0x7030, 0x02, 1}, // Reset + {0x7030, 0x03, 1}, // Reduce torque + {0x0000, 0x00, 5}, // Gap + {0x0000, 0x00, 8} // Gap +}; + +static ec_pdo_entry_info_t lcec_el7342_channel2_vel_out[] = { + {0x7030, 0x21, 16} // Velocity +}; + +static ec_pdo_entry_info_t lcec_el7342_channel1_enc_in[] = { + {0x0000, 0x00, 1}, // Gap + {0x6000, 0x02, 1}, // Latch extern valid + {0x6000, 0x03, 1}, // Set counter done + {0x6000, 0x04, 1}, // Counter underflow + {0x6000, 0x05, 1}, // Counter overflow + {0x0000, 0x00, 2}, // Gap + {0x6000, 0x08, 1}, // Extrapolation stall + {0x6000, 0x09, 1}, // Status of input A + {0x6000, 0x0a, 1}, // Status of input B + {0x0000, 0x00, 1}, // Gap + {0x0000, 0x00, 1}, // Gap + {0x6000, 0x0d, 1}, // Status of extern latch + {0x1c32, 0x20, 1}, // Sync error + {0x0000, 0x00, 1}, // Gap + {0x1800, 0x09, 1}, // TxPDO Toggle + {0x6000, 0x11, 16}, // Counter value + {0x6000, 0x12, 16} // Latch value +}; + +static ec_pdo_entry_info_t lcec_el7342_channel2_enc_in[] = { + {0x0000, 0x00, 1}, // Gap + {0x6010, 0x02, 1}, // Latch extern valid + {0x6010, 0x03, 1}, // Set counter done + {0x6010, 0x04, 1}, // Counter underflow + {0x6010, 0x05, 1}, // Counter overflow + {0x0000, 0x00, 2}, // Gap + {0x6010, 0x08, 1}, // Extrapolation stall + {0x6010, 0x09, 1}, // Status of input A + {0x6010, 0x0a, 1}, // Status of input B + {0x0000, 0x00, 1}, // Gap + {0x0000, 0x00, 1}, // Gap + {0x6010, 0x0d, 1}, // Status of extern latch + {0x1c32, 0x20, 1}, // Sync error + {0x0000, 0x00, 1}, // Gap + {0x1803, 0x09, 1}, // TxPDO Toggle + {0x6010, 0x11, 16}, // Counter value + {0x6010, 0x12, 16} // Latch value +}; + +static ec_pdo_entry_info_t lcec_el7342_channel1_dcm_in[] = { + {0x6020, 0x01, 1}, // Ready to enable + {0x6020, 0x02, 1}, // Ready + {0x6020, 0x03, 1}, // Warning + {0x6020, 0x04, 1}, // Error + {0x6020, 0x05, 1}, // Moving positive + {0x6020, 0x06, 1}, // Moving negative + {0x6020, 0x07, 1}, // Torque reduced + {0x0000, 0x00, 1}, // Gap + {0x0000, 0x00, 3}, // Gap + {0x6020, 0x0c, 1}, // Digital input 1 + {0x6020, 0x0d, 1}, // Digital input 2 + {0x1c32, 0x20, 1}, // Sync error + {0x0000, 0x00, 1}, // Gap + {0x1806, 0x09, 1} // TxPDO Toggle +}; + +static ec_pdo_entry_info_t lcec_el7342_channel1_dcm_sync_info[] = { + {0x6020, 0x11, 16}, // Synchronous information 1 + {0x6020, 0x12, 16} // Synchronous information 2 +}; + +static ec_pdo_entry_info_t lcec_el7342_channel2_dcm_in[] = { + {0x6030, 0x01, 1}, // Ready to enable + {0x6030, 0x02, 1}, // Ready + {0x6030, 0x03, 1}, // Warning + {0x6030, 0x04, 1}, // Error + {0x6030, 0x05, 1}, // Moving positive + {0x6030, 0x06, 1}, // Moving negative + {0x6030, 0x07, 1}, // Torque reduced + {0x0000, 0x00, 1}, // Gap + {0x0000, 0x00, 3}, // Gap + {0x6030, 0x0c, 1}, // Digital input 1 + {0x6030, 0x0d, 1}, // Digital input 2 + {0x1c32, 0x20, 1}, // Sync error + {0x0000, 0x00, 1}, // Gap + {0x1808, 0x09, 1} // TxPDO Toggle +}; + +static ec_pdo_entry_info_t lcec_el7342_channel2_dcm_sync_info[] = { + {0x6030, 0x11, 16}, // Synchronous information 1 + {0x6030, 0x12, 16} // Synchronous information 2 +}; + +static ec_pdo_info_t lcec_el7342_pdos_out[] = { + {0x1600, 7, lcec_el7342_channel1_enc_out}, + {0x1602, 7, lcec_el7342_channel2_enc_out}, + {0x1604, 5, lcec_el7342_channel1_dcm_out}, + {0x1606, 1, lcec_el7342_channel1_vel_out}, + {0x1607, 5, lcec_el7342_channel2_dcm_out}, + {0x1609, 1, lcec_el7342_channel2_vel_out} +}; + +static ec_pdo_info_t lcec_el7342_pdos_in[] = { + {0x1a00, 17, lcec_el7342_channel1_enc_in}, + {0x1a03, 17, lcec_el7342_channel2_enc_in}, + {0x1a06, 14, lcec_el7342_channel1_dcm_in}, + {0x1a07, 2, lcec_el7342_channel1_dcm_sync_info}, + {0x1a08, 14, lcec_el7342_channel2_dcm_in}, + {0x1a09, 2, lcec_el7342_channel2_dcm_sync_info}, +}; + +static ec_sync_info_t lcec_el7342_syncs[] = { + {0, EC_DIR_OUTPUT, 0, NULL}, + {1, EC_DIR_INPUT, 0, NULL}, + {2, EC_DIR_OUTPUT, 6, lcec_el7342_pdos_out}, + {3, EC_DIR_INPUT, 6, lcec_el7342_pdos_in}, + {0xff} +}; + +void lcec_el7342_read(struct lcec_slave *slave, long period); +void lcec_el7342_write(struct lcec_slave *slave, long period); + +void lcec_el7342_set_info(lcec_el7342_chan_t *chan, hal_s32_t *raw_info, hal_u32_t *sel_info); + +int lcec_el7342_init(int comp_id, struct lcec_slave *slave, ec_pdo_entry_reg_t *pdo_entry_regs) { + lcec_master_t *master = slave->master; + lcec_el7342_data_t *hal_data; + int i; + lcec_el7342_chan_t *chan; + uint8_t info1_select, info2_select; + int err; + + // initialize callbacks + slave->proc_read = lcec_el7342_read; + slave->proc_write = lcec_el7342_write; + + // alloc hal memory + if ((hal_data = hal_malloc(sizeof(lcec_el7342_data_t))) == NULL) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "hal_malloc() for slave %s.%s failed\n", master->name, slave->name); + return -EIO; + } + memset(hal_data, 0, sizeof(lcec_el7342_data_t)); + slave->hal_data = hal_data; + + // initialize sync info + slave->sync_info = lcec_el7342_syncs; + + // initialize global data + hal_data->last_operational = 0; + + // initialize pins + for (i=0; ichans[i]; + + // read sdos + // Info1 selector + if ((chan->sdo_info1_select = lcec_read_sdo(slave, 0x8022 + (i << 4), 0x11, 1)) == NULL) { + return -EIO; + } + info1_select = EC_READ_U8(ecrt_sdo_request_data(chan->sdo_info1_select)); + // Info2 selector + if ((chan->sdo_info2_select = lcec_read_sdo(slave, 0x8022 + (i << 4), 0x19, 1)) == NULL) { + return -EIO; + } + info2_select = EC_READ_U8(ecrt_sdo_request_data(chan->sdo_info2_select)); + + // initialize POD entries + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6000 + (i << 4), 0x02, &chan->latch_ext_valid_pdo_os, &chan->latch_ext_valid_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6000 + (i << 4), 0x03, &chan->set_count_done_pdo_os, &chan->set_count_done_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6000 + (i << 4), 0x04, &chan->count_overflow_pdo_os, &chan->count_overflow_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6000 + (i << 4), 0x05, &chan->count_underflow_pdo_os, &chan->count_underflow_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6000 + (i << 4), 0x08, &chan->expol_stall_pdo_os, &chan->expol_stall_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6000 + (i << 4), 0x09, &chan->ina_pdo_os, &chan->ina_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6000 + (i << 4), 0x0a, &chan->inb_pdo_os, &chan->inb_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6000 + (i << 4), 0x0d, &chan->inext_pdo_os, &chan->inext_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x1c32 , 0x20, &chan->sync_err_pdo_os, &chan->sync_err_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x1800 + (i * 3), 0x09, &chan->tx_toggle_pdo_os, &chan->tx_toggle_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6000 + (i << 4), 0x11, &chan->count_pdo_os, NULL); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6000 + (i << 4), 0x12, &chan->latch_pdo_os, NULL); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x7000 + (i << 4), 0x02, &chan->ena_latch_ext_pos_pdo_os, &chan->ena_latch_ext_pos_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x7000 + (i << 4), 0x03, &chan->set_count_pdo_os, &chan->set_count_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x7000 + (i << 4), 0x04, &chan->ena_latch_ext_neg_pdo_os, &chan->ena_latch_ext_neg_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x7000 + (i << 4), 0x11, &chan->set_count_val_pdo_os, NULL); + + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6020 + (i << 4), 0x01, &chan->dcm_ready_to_enable_pdo_os, &chan->dcm_ready_to_enable_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6020 + (i << 4), 0x02, &chan->dcm_ready_pdo_os, &chan->dcm_ready_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6020 + (i << 4), 0x03, &chan->dcm_warning_pdo_os, &chan->dcm_warning_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6020 + (i << 4), 0x04, &chan->dcm_error_pdo_os, &chan->dcm_error_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6020 + (i << 4), 0x05, &chan->dcm_move_pos_pdo_os, &chan->dcm_move_pos_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6020 + (i << 4), 0x06, &chan->dcm_move_neg_pdo_os, &chan->dcm_move_neg_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6020 + (i << 4), 0x07, &chan->dcm_torque_reduced_pdo_os, &chan->dcm_torque_reduced_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6020 + (i << 4), 0x0c, &chan->dcm_din1_pdo_os, &chan->dcm_din1_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6020 + (i << 4), 0x0d, &chan->dcm_din2_pdo_os, &chan->dcm_din2_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x1c32 , 0x20, &chan->dcm_sync_err_pdo_os, &chan->dcm_sync_err_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x1806 + (i << 1), 0x09, &chan->dcm_tx_toggle_pdo_os, &chan->dcm_tx_toggle_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6020 + (i << 4), 0x11, &chan->dcm_info1_pdo_os, NULL); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6020 + (i << 4), 0x12, &chan->dcm_info2_pdo_os, NULL); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x7020 + (i << 4), 0x01, &chan->dcm_ena_pdo_os, &chan->dcm_ena_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x7020 + (i << 4), 0x02, &chan->dcm_reset_pdo_os, &chan->dcm_reset_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x7020 + (i << 4), 0x03, &chan->dcm_reduce_torque_pdo_os, &chan->dcm_reduce_torque_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x7020 + (i << 4), 0x21, &chan->dcm_velo_pdo_os, NULL); + + // export encoder pins + if ((err = hal_pin_bit_newf(HAL_IN, &(chan->reset), comp_id, "%s.%s.%s.enc-%d-reset", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-reset failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(chan->ina), comp_id, "%s.%s.%s.enc-%d-ina", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-ina failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(chan->inb), comp_id, "%s.%s.%s.enc-%d-inb", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-inb failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(chan->inext), comp_id, "%s.%s.%s.enc-%d-inext", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-inext failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(chan->sync_err), comp_id, "%s.%s.%s.enc-%d-sync-error", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-sync-error failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(chan->expol_stall), comp_id, "%s.%s.%s.enc-%d-expol-stall", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-expol-stall failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(chan->tx_toggle), comp_id, "%s.%s.%s.enc-%d-tx-toggle", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-tx-toggle failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(chan->count_overflow), comp_id, "%s.%s.%s.enc-%d-count-overflow", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-count-overflow failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(chan->count_underflow), comp_id, "%s.%s.%s.enc-%d-count-underflow", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-count-underflow failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(chan->latch_ext_valid), comp_id, "%s.%s.%s.enc-%d-latch-ext-valid", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-latch-ext-valid failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_IO, &(chan->set_raw_count), comp_id, "%s.%s.%s.enc-%d-set-raw-count", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-set-raw-count failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_IO, &(chan->ena_latch_ext_pos), comp_id, "%s.%s.%s.enc-%d-index-ext-pos-enable", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-ndex-ext-pos-enable failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_IO, &(chan->ena_latch_ext_neg), comp_id, "%s.%s.%s.enc-%d-index-ext-neg-enable", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-ndex-ext-neg-enable failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_s32_newf(HAL_IN, &(chan->set_raw_count_val), comp_id, "%s.%s.%s.enc-%d-set-raw-count-val", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-set-raw-count-val failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_s32_newf(HAL_OUT, &(chan->raw_count), comp_id, "%s.%s.%s.enc-%d-raw-count", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-raw-count failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_s32_newf(HAL_OUT, &(chan->count), comp_id, "%s.%s.%s.enc-%d-count", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-count failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_s32_newf(HAL_OUT, &(chan->raw_latch), comp_id, "%s.%s.%s.enc-%d-raw-latch", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-raw-latch failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_float_newf(HAL_OUT, &(chan->pos), comp_id, "%s.%s.%s.enc-%d-pos", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-pos failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_float_newf(HAL_IO, &(chan->pos_scale), comp_id, "%s.%s.%s.enc-%d-pos-scale", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-scale failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + + // export servo pins + if ((err = hal_pin_float_newf(HAL_IO, &(chan->dcm_scale), comp_id, "%s.%s.%s.srv-%d-scale", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-%d-scale failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_float_newf(HAL_IO, &(chan->dcm_offset), comp_id, "%s.%s.%s.srv-%d-offset", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-%d-offset failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_float_newf(HAL_IO, &(chan->dcm_min_dc), comp_id, "%s.%s.%s.srv-%d-min-dc", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-%d-min-dc failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_float_newf(HAL_IO, &(chan->dcm_max_dc), comp_id, "%s.%s.%s.srv-%d-max-dc", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-%d-max-dc failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_float_newf(HAL_OUT, &(chan->dcm_curr_dc), comp_id, "%s.%s.%s.srv-%d-curr-dc", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-%d-curr-dc failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_IN, &(chan->dcm_enable), comp_id, "%s.%s.%s.srv-%d-enable", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-%d-enable failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_IN, &(chan->dcm_absmode), comp_id, "%s.%s.%s.srv-%d-absmode", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-%d-absmode failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_float_newf(HAL_IN, &(chan->dcm_value), comp_id, "%s.%s.%s.srv-%d-cmd", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-%d-value failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_s32_newf(HAL_OUT, &(chan->dcm_raw_val), comp_id, "%s.%s.%s.srv-%d-raw-cmd", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-%d-raw-cmd failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_IN, &(chan->dcm_reset), comp_id, "%s.%s.%s.srv-%d-reset", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-%d-reset failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_IN, &(chan->dcm_reduce_torque), comp_id, "%s.%s.%s.srv-%d-reduce-torque", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-%d-reduce-torque failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(chan->dcm_ready_to_enable), comp_id, "%s.%s.%s.srv-%d-ready-to-enable", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-ready-to-enable%d- failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(chan->dcm_ready), comp_id, "%s.%s.%s.srv-%d-ready", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-ready%d- failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(chan->dcm_warning), comp_id, "%s.%s.%s.srv-%d-warning", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-%d-warning failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(chan->dcm_error), comp_id, "%s.%s.%s.srv-%d-error", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-%d-error failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(chan->dcm_move_pos), comp_id, "%s.%s.%s.srv-%d-move-pos", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-%d-move-pos failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(chan->dcm_move_neg), comp_id, "%s.%s.%s.srv-%d-move-neg", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-%d-move-neg failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(chan->dcm_torque_reduced), comp_id, "%s.%s.%s.srv-%d-torque-reduced", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-%d-torque-reduced failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(chan->dcm_din1), comp_id, "%s.%s.%s.srv-%d-din1", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-%d-din1 failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(chan->dcm_din2), comp_id, "%s.%s.%s.srv-%d-din2", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-%d-din2 failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(chan->dcm_sync_err), comp_id, "%s.%s.%s.srv-%d-sync-error", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-%d-sync-error failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(chan->dcm_tx_toggle), comp_id, "%s.%s.%s.srv-%d-tx-toggle", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-%d-tx-toggle failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_s32_newf(HAL_OUT, &(chan->dcm_raw_info1), comp_id, "%s.%s.%s.srv-%d-raw-info1", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-%d-raw-info1 failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_s32_newf(HAL_OUT, &(chan->dcm_raw_info2), comp_id, "%s.%s.%s.srv-%d-raw-info2", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-%d-raw-info2 failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_u32_newf(HAL_OUT, &(chan->dcm_sel_info1), comp_id, "%s.%s.%s.srv-%d-sel-info1", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-%d-sel-info1 failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_u32_newf(HAL_OUT, &(chan->dcm_sel_info2), comp_id, "%s.%s.%s.srv-%d-sel-info2", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-%d-sel-info2 failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if (info1_select == INFO_SEL_MOTOR_VELO || info2_select == INFO_SEL_MOTOR_VELO) { + if ((err = hal_pin_float_newf(HAL_OUT, &(chan->dcm_velo_fb), comp_id, "%s.%s.%s.srv-%d-velo-fb", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-velo-fb failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + } + if (info1_select == INFO_SEL_MOTOR_CURR || info2_select == INFO_SEL_MOTOR_CURR) { + if ((err = hal_pin_float_newf(HAL_OUT, &(chan->dcm_current_fb), comp_id, "%s.%s.%s.srv-%d-current-fb", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-current-fb failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + } + + // initialize pins + *(chan->reset) = 0; + *(chan->ina) = 0; + *(chan->inb) = 0; + *(chan->inext) = 0; + *(chan->sync_err) = 0; + *(chan->expol_stall) = 0; + *(chan->tx_toggle) = 0; + *(chan->count_overflow) = 0; + *(chan->count_underflow) = 0; + *(chan->latch_ext_valid) = 0; + *(chan->ena_latch_ext_pos) = 0; + *(chan->ena_latch_ext_neg) = 0; + *(chan->set_raw_count) = 0; + *(chan->set_raw_count_val) = 0; + *(chan->raw_count) = 0; + *(chan->raw_latch) = 0; + *(chan->count) = 0; + *(chan->pos) = 0; + *(chan->pos_scale) = 1.0; + + *(chan->dcm_scale) = 1.0; + *(chan->dcm_offset) = 0.0; + *(chan->dcm_min_dc) = -1.0; + *(chan->dcm_max_dc) = 1.0; + *(chan->dcm_curr_dc) = 0.0; + *(chan->dcm_enable) = 0; + *(chan->dcm_absmode) = 0; + *(chan->dcm_value) = 0.0; + *(chan->dcm_raw_val) = 0; + *(chan->dcm_reset) = 0; + *(chan->dcm_reduce_torque) = 0; + *(chan->dcm_ready_to_enable) = 0; + *(chan->dcm_ready) = 0; + *(chan->dcm_warning) = 0; + *(chan->dcm_error) = 0; + *(chan->dcm_move_pos) = 0; + *(chan->dcm_move_neg) = 0; + *(chan->dcm_torque_reduced) = 0; + *(chan->dcm_din1) = 0; + *(chan->dcm_din2) = 0; + *(chan->dcm_sync_err) = 0; + *(chan->dcm_tx_toggle) = 0; + *(chan->dcm_raw_info1) = 0; + *(chan->dcm_raw_info2) = 0; + *(chan->dcm_sel_info1) = info1_select; + *(chan->dcm_sel_info2) = info2_select; + if (chan->dcm_velo_fb != NULL) { + *(chan->dcm_velo_fb) = 0.0; + } + if (chan->dcm_current_fb != NULL) { + *(chan->dcm_current_fb) = 0.0; + } + + // initialize variables + chan->enc_do_init = 1; + chan->enc_last_count = 0; + chan->enc_old_scale = *(chan->pos_scale) + 1.0; + chan->enc_scale_recip = 1.0; + + chan->dcm_old_scale = *(chan->dcm_scale) + 1.0; + chan->dcm_scale_recip = 1.0; + } + + return 0; +} + +void lcec_el7342_read(struct lcec_slave *slave, long period) { + lcec_master_t *master = slave->master; + lcec_el7342_data_t *hal_data = (lcec_el7342_data_t *) slave->hal_data; + uint8_t *pd = master->process_data; + int i; + lcec_el7342_chan_t *chan; + int16_t raw_count, raw_latch, raw_delta; + + // wait for slave to be operational + if (!slave->state.operational) { + hal_data->last_operational = 0; + return; + } + + // check inputs + for (i=0; ichans[i]; + + // check for change in scale value + if (*(chan->pos_scale) != chan->enc_old_scale) { + // scale value has changed, test and update it + if ((*(chan->pos_scale) < 1e-20) && (*(chan->pos_scale) > -1e-20)) { + // value too small, divide by zero is a bad thing + *(chan->pos_scale) = 1.0; + } + // save new scale to detect future changes + chan->enc_old_scale = *(chan->pos_scale); + // we actually want the reciprocal + chan->enc_scale_recip = 1.0 / *(chan->pos_scale); + } + + // get bit states + *(chan->ina) = EC_READ_BIT(&pd[chan->ina_pdo_os], chan->ina_pdo_bp); + *(chan->inb) = EC_READ_BIT(&pd[chan->inb_pdo_os], chan->inb_pdo_bp); + *(chan->inext) = EC_READ_BIT(&pd[chan->inext_pdo_os], chan->inext_pdo_bp); + *(chan->sync_err) = EC_READ_BIT(&pd[chan->sync_err_pdo_os], chan->sync_err_pdo_bp); + *(chan->expol_stall) = EC_READ_BIT(&pd[chan->expol_stall_pdo_os], chan->expol_stall_pdo_bp); + *(chan->tx_toggle) = EC_READ_BIT(&pd[chan->tx_toggle_pdo_os], chan->tx_toggle_pdo_bp); + *(chan->count_overflow) = EC_READ_BIT(&pd[chan->count_overflow_pdo_os], chan->count_overflow_pdo_bp); + *(chan->count_underflow) = EC_READ_BIT(&pd[chan->count_underflow_pdo_os], chan->count_underflow_pdo_bp); + *(chan->latch_ext_valid) = EC_READ_BIT(&pd[chan->latch_ext_valid_pdo_os], chan->latch_ext_valid_pdo_bp); + + *(chan->dcm_ready_to_enable) = EC_READ_BIT(&pd[chan->dcm_ready_to_enable_pdo_os], chan->dcm_ready_to_enable_pdo_bp); + *(chan->dcm_ready) = EC_READ_BIT(&pd[chan->dcm_ready_pdo_os], chan->dcm_ready_pdo_bp); + *(chan->dcm_warning) = EC_READ_BIT(&pd[chan->dcm_warning_pdo_os], chan->dcm_warning_pdo_bp); + *(chan->dcm_error) = EC_READ_BIT(&pd[chan->dcm_error_pdo_os], chan->dcm_error_pdo_bp); + *(chan->dcm_move_pos) = EC_READ_BIT(&pd[chan->dcm_move_pos_pdo_os], chan->dcm_move_pos_pdo_bp); + *(chan->dcm_move_neg) = EC_READ_BIT(&pd[chan->dcm_move_neg_pdo_os], chan->dcm_move_neg_pdo_bp); + *(chan->dcm_torque_reduced) = EC_READ_BIT(&pd[chan->dcm_torque_reduced_pdo_os], chan->dcm_torque_reduced_pdo_bp); + *(chan->dcm_din1) = EC_READ_BIT(&pd[chan->dcm_din1_pdo_os], chan->dcm_din1_pdo_bp); + *(chan->dcm_din2) = EC_READ_BIT(&pd[chan->dcm_din2_pdo_os], chan->dcm_din2_pdo_bp); + *(chan->dcm_sync_err) = EC_READ_BIT(&pd[chan->dcm_sync_err_pdo_os], chan->dcm_sync_err_pdo_bp); + *(chan->dcm_tx_toggle) = EC_READ_BIT(&pd[chan->dcm_tx_toggle_pdo_os], chan->dcm_tx_toggle_pdo_bp); + + // read raw values + raw_count = EC_READ_S16(&pd[chan->count_pdo_os]); + raw_latch = EC_READ_S16(&pd[chan->latch_pdo_os]); + + // read raw info values + *(chan->dcm_raw_info1) = EC_READ_S16(&pd[chan->dcm_info1_pdo_os]); + *(chan->dcm_raw_info2) = EC_READ_S16(&pd[chan->dcm_info2_pdo_os]); + + // dispatch info values + lcec_el7342_set_info(chan, chan->dcm_raw_info1, chan->dcm_sel_info1); + lcec_el7342_set_info(chan, chan->dcm_raw_info2, chan->dcm_sel_info2); + + // check for operational change of slave + if (!hal_data->last_operational) { + chan->enc_last_count = raw_count; + } + + // check for counter set done + if (EC_READ_BIT(&pd[chan->set_count_done_pdo_os], chan->set_count_done_pdo_bp)) { + chan->enc_last_count = raw_count; + *(chan->set_raw_count) = 0; + } + + // update raw values + if (! *(chan->set_raw_count)) { + *(chan->raw_count) = raw_count; + } + + // handle initialization + if (chan->enc_do_init || *(chan->reset)) { + chan->enc_do_init = 0; + chan->enc_last_count = raw_count; + *(chan->count) = 0; + } + + // handle index + if (*(chan->latch_ext_valid)) { + *(chan->raw_latch) = raw_latch; + chan->enc_last_count = raw_latch; + *(chan->count) = 0; + *(chan->ena_latch_ext_pos) = 0; + *(chan->ena_latch_ext_neg) = 0; + } + + // compute net counts + raw_delta = raw_count - chan->enc_last_count; + chan->enc_last_count = raw_count; + *(chan->count) += raw_delta; + + // scale count to make floating point position + *(chan->pos) = *(chan->count) * chan->enc_scale_recip; + } + + hal_data->last_operational = 1; +} + +void lcec_el7342_write(struct lcec_slave *slave, long period) { + lcec_master_t *master = slave->master; + lcec_el7342_data_t *hal_data = (lcec_el7342_data_t *) slave->hal_data; + uint8_t *pd = master->process_data; + int i; + lcec_el7342_chan_t *chan; + double tmpval, tmpdc, raw_val; + + // set outputs + for (i=0; ichans[i]; + + // validate duty cycle limits, both limits must be between + // 0.0 and 1.0 (inclusive) and max must be greater then min + if (*(chan->dcm_max_dc) > 1.0) { + *(chan->dcm_max_dc) = 1.0; + } + if (*(chan->dcm_min_dc) > *(chan->dcm_max_dc)) { + *(chan->dcm_min_dc) = *(chan->dcm_max_dc); + } + if (*(chan->dcm_min_dc) < -1.0) { + *(chan->dcm_min_dc) = -1.0; + } + if (*(chan->dcm_max_dc) < *(chan->dcm_min_dc)) { + *(chan->dcm_max_dc) = *(chan->dcm_min_dc); + } + + // do scale calcs only when scale changes + if (*(chan->dcm_scale) != chan->dcm_old_scale) { + // validate the new scale value + if ((*(chan->dcm_scale) < 1e-20) && (*(chan->dcm_scale) > -1e-20)) { + // value too small, divide by zero is a bad thing + *(chan->dcm_scale) = 1.0; + } + // get ready to detect future scale changes + chan->dcm_old_scale = *(chan->dcm_scale); + // we will need the reciprocal + chan->dcm_scale_recip = 1.0 / *(chan->dcm_scale); + } + + // get command + tmpval = *(chan->dcm_value); + if (*(chan->dcm_absmode) && (tmpval < 0)) { + tmpval = -tmpval; + } + + // convert value command to duty cycle + tmpdc = tmpval * chan->dcm_scale_recip + *(chan->dcm_offset); + if (tmpdc < *(chan->dcm_min_dc)) { + tmpdc = *(chan->dcm_min_dc); + } + if (tmpdc > *(chan->dcm_max_dc)) { + tmpdc = *(chan->dcm_max_dc); + } + + // set output values + if (*(chan->dcm_enable) == 0) { + raw_val = 0; + *(chan->dcm_curr_dc) = 0; + } else { + raw_val = (double)0x7fff * tmpdc; + if (raw_val > (double)0x7fff) { + raw_val = (double)0x7fff; + } + if (raw_val < (double)-0x7fff) { + raw_val = (double)-0x7fff; + } + *(chan->dcm_curr_dc) = tmpdc; + } + + // update value + *(chan->dcm_raw_val) = (int32_t)raw_val; + + // set output data + EC_WRITE_BIT(&pd[chan->set_count_pdo_os], chan->set_count_pdo_bp, *(chan->set_raw_count)); + EC_WRITE_BIT(&pd[chan->ena_latch_ext_pos_pdo_os], chan->ena_latch_ext_pos_pdo_bp, *(chan->ena_latch_ext_pos)); + EC_WRITE_BIT(&pd[chan->ena_latch_ext_neg_pdo_os], chan->ena_latch_ext_neg_pdo_bp, *(chan->ena_latch_ext_neg)); + EC_WRITE_S16(&pd[chan->set_count_val_pdo_os], *(chan->set_raw_count_val)); + + EC_WRITE_BIT(&pd[chan->dcm_ena_pdo_os], chan->dcm_ena_pdo_bp, *(chan->dcm_enable)); + EC_WRITE_BIT(&pd[chan->dcm_reset_pdo_os], chan->dcm_reset_pdo_bp, *(chan->dcm_reset)); + EC_WRITE_BIT(&pd[chan->dcm_reduce_torque_pdo_os], chan->dcm_reduce_torque_pdo_bp, *(chan->dcm_reduce_torque)); + EC_WRITE_S16(&pd[chan->dcm_velo_pdo_os], (int16_t)raw_val); + } +} + +void lcec_el7342_set_info(lcec_el7342_chan_t *chan, hal_s32_t *raw_info, hal_u32_t *sel_info) { + switch(*sel_info) { + case INFO_SEL_MOTOR_VELO: + *(chan->dcm_velo_fb) = (double) *raw_info * 0.0001 * chan->dcm_old_scale; + break; + + case INFO_SEL_MOTOR_CURR: + *(chan->dcm_current_fb) = (double) *raw_info * 0.001; + break; + } +} + diff --git a/src/lcec_el7342.h b/src/lcec_el7342.h new file mode 100644 index 00000000..1419b6b7 --- /dev/null +++ b/src/lcec_el7342.h @@ -0,0 +1,32 @@ +// +// Copyright (C) 2011 Sascha Ittner +// +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2 of the License, or +// (at your option) any later version. +// +// This program 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 this program; if not, write to the Free Software +// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA +// +#ifndef _LCEC_EL7342_H_ +#define _LCEC_EL7342_H_ + +#include "lcec.h" + +#define LCEC_EL7342_VID LCEC_BECKHOFF_VID +#define LCEC_EL7342_PID 0x1cae3052 + +#define LCEC_EL7342_CHANS 2 +#define LCEC_EL7342_PDOS (33 * LCEC_EL7342_CHANS) + +int lcec_el7342_init(int comp_id, struct lcec_slave *slave, ec_pdo_entry_reg_t *pdo_entry_regs); + +#endif + diff --git a/src/lcec_el95xx.c b/src/lcec_el95xx.c new file mode 100644 index 00000000..22c6332c --- /dev/null +++ b/src/lcec_el95xx.c @@ -0,0 +1,84 @@ +// +// Copyright (C) 2011 Sascha Ittner +// +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2 of the License, or +// (at your option) any later version. +// +// This program 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 this program; if not, write to the Free Software +// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA +// + +#include "lcec.h" +#include "lcec_el95xx.h" + +typedef struct { + hal_bit_t *power_ok; + hal_bit_t *overload; + unsigned int power_ok_pdo_os; + unsigned int power_ok_pdo_bp; + unsigned int overload_pdo_os; + unsigned int overload_pdo_bp; +} lcec_el95xx_data_t; + +void lcec_el95xx_read(struct lcec_slave *slave, long period); + +int lcec_el95xx_init(int comp_id, struct lcec_slave *slave, ec_pdo_entry_reg_t *pdo_entry_regs) { + lcec_master_t *master = slave->master; + lcec_el95xx_data_t *hal_data; + int err; + + // initialize callbacks + slave->proc_read = lcec_el95xx_read; + + // alloc hal memory + if ((hal_data = hal_malloc(sizeof(lcec_el95xx_data_t))) == NULL) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "hal_malloc() for slave %s.%s failed\n", master->name, slave->name); + return -EIO; + } + memset(hal_data, 0, sizeof(lcec_el95xx_data_t)); + slave->hal_data = hal_data; + + // initialize POD entries + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6000, 0x01, &hal_data->power_ok_pdo_os, &hal_data->power_ok_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6000, 0x02, &hal_data->overload_pdo_os, &hal_data->overload_pdo_bp); + + // export pins + if ((err = hal_pin_bit_newf(HAL_OUT, &(hal_data->power_ok), comp_id, "%s.%s.%s.power-ok", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.power-ok failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(hal_data->overload), comp_id, "%s.%s.%s.overload", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.overload failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + + // initialize pins + *(hal_data->power_ok) = 0; + *(hal_data->overload) = 0; + + return 0; +} + +void lcec_el95xx_read(struct lcec_slave *slave, long period) { + lcec_master_t *master = slave->master; + lcec_el95xx_data_t *hal_data = (lcec_el95xx_data_t *) slave->hal_data; + uint8_t *pd = master->process_data; + + // wait for slave to be operational + if (!slave->state.operational) { + return; + } + + // check inputs + *(hal_data->power_ok) = EC_READ_BIT(&pd[hal_data->power_ok_pdo_os], hal_data->power_ok_pdo_bp); + *(hal_data->overload) = EC_READ_BIT(&pd[hal_data->overload_pdo_os], hal_data->overload_pdo_bp); +} + diff --git a/src/lcec_el95xx.h b/src/lcec_el95xx.h new file mode 100644 index 00000000..137f3c2d --- /dev/null +++ b/src/lcec_el95xx.h @@ -0,0 +1,36 @@ +// +// Copyright (C) 2011 Sascha Ittner +// +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2 of the License, or +// (at your option) any later version. +// +// This program 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 this program; if not, write to the Free Software +// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA +// +#ifndef _LCEC_EL95XX_H_ +#define _LCEC_EL95XX_H_ + +#include "lcec.h" + +#define LCEC_EL95xx_VID LCEC_BECKHOFF_VID + +#define LCEC_EL9505_PID 0x25213052 +#define LCEC_EL9508_PID 0x25243052 +#define LCEC_EL9510_PID 0x25263052 +#define LCEC_EL9512_PID 0x25283052 +#define LCEC_EL9515_PID 0x252b3052 + +#define LCEC_EL95xx_PDOS 2 + +int lcec_el95xx_init(int comp_id, struct lcec_slave *slave, ec_pdo_entry_reg_t *pdo_entry_regs); + +#endif + diff --git a/src/lcec_em7004.c b/src/lcec_em7004.c new file mode 100644 index 00000000..8f278637 --- /dev/null +++ b/src/lcec_em7004.c @@ -0,0 +1,540 @@ +// +// Copyright (C) 2014 Sascha Ittner +// +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2 of the License, or +// (at your option) any later version. +// +// This program 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 this program; if not, write to the Free Software +// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA +// + +#include "lcec.h" +#include "lcec_em7004.h" + +typedef struct { + hal_bit_t *in; + hal_bit_t *in_not; + unsigned int pdo_os; + unsigned int pdo_bp; +} lcec_em7004_din_t; + +typedef struct { + hal_bit_t *out; + hal_bit_t invert; + unsigned int pdo_os; + unsigned int pdo_bp; +} lcec_em7004_dout_t; + +typedef struct { + hal_bit_t *pos; + hal_bit_t *neg; + hal_bit_t *enable; + hal_bit_t *absmode; + hal_float_t *value; + hal_float_t *scale; + hal_float_t *offset; + double old_scale; + double scale_recip; + hal_float_t *min_dc; + hal_float_t *max_dc; + hal_float_t *curr_dc; + hal_s32_t *raw_val; + unsigned int val_pdo_os; +} lcec_em7004_aout_t; + +typedef struct { + hal_bit_t *ena_latch_ext_pos; + hal_bit_t *ena_latch_ext_neg; + hal_bit_t *reset; + hal_bit_t *ina; + hal_bit_t *inb; + hal_bit_t *ingate; + hal_bit_t *inext; + hal_bit_t *latch_ext_valid; + hal_bit_t *set_raw_count; + hal_s32_t *set_raw_count_val; + hal_s32_t *raw_count; + hal_s32_t *raw_latch; + hal_s32_t *count; + hal_float_t *pos_scale; + hal_float_t *pos; + + unsigned int ena_latch_ext_pos_pdo_os; + unsigned int ena_latch_ext_pos_pdo_bp; + unsigned int ena_latch_ext_neg_pdo_os; + unsigned int ena_latch_ext_neg_pdo_bp; + unsigned int set_count_pdo_os; + unsigned int set_count_pdo_bp; + unsigned int set_count_val_pdo_os; + unsigned int set_count_done_pdo_os; + unsigned int set_count_done_pdo_bp; + unsigned int latch_ext_valid_pdo_os; + unsigned int latch_ext_valid_pdo_bp; + unsigned int ina_pdo_os; + unsigned int ina_pdo_bp; + unsigned int inb_pdo_os; + unsigned int inb_pdo_bp; + unsigned int ingate_pdo_os; + unsigned int ingate_pdo_bp; + unsigned int inext_pdo_os; + unsigned int inext_pdo_bp; + unsigned int count_pdo_os; + unsigned int latch_pdo_os; + + int do_init; + int16_t last_count; + double old_scale; + double scale; +} lcec_em7004_enc_t; + +typedef struct { + lcec_em7004_din_t dins[LCEC_EM7004_DIN_COUNT]; + lcec_em7004_dout_t douts[LCEC_EM7004_DOUT_COUNT]; + lcec_em7004_aout_t aouts[LCEC_EM7004_AOUT_COUNT]; + lcec_em7004_enc_t encs[LCEC_EM7004_ENC_COUNT]; + int last_operational; +} lcec_em7004_data_t; + +void lcec_em7004_read(struct lcec_slave *slave, long period); +void lcec_em7004_write(struct lcec_slave *slave, long period); + +int lcec_em7004_init(int comp_id, struct lcec_slave *slave, ec_pdo_entry_reg_t *pdo_entry_regs) { + lcec_master_t *master = slave->master; + lcec_em7004_data_t *hal_data; + lcec_em7004_din_t *din; + lcec_em7004_dout_t *dout; + lcec_em7004_aout_t *aout; + lcec_em7004_enc_t *enc; + int i; + int err; + + // initialize callbacks + slave->proc_read = lcec_em7004_read; + slave->proc_write = lcec_em7004_write; + + // alloc hal memory + if ((hal_data = hal_malloc(sizeof(lcec_em7004_data_t))) == NULL) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "hal_malloc() for slave %s.%s failed\n", master->name, slave->name); + return -EIO; + } + memset(hal_data, 0, sizeof(lcec_em7004_data_t)); + slave->hal_data = hal_data; + + // initialize global data + hal_data->last_operational = 0; + + // initialize digital input pins + for (i=0, din=hal_data->dins; iindex, slave->vid, slave->pid, 0x6000, i + 1, &din->pdo_os, &din->pdo_bp); + + // export dins + if ((err = hal_pin_bit_newf(HAL_OUT, &(din->in), comp_id, "%s.%s.%s.din-%d", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.din-%02d failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(din->in_not), comp_id, "%s.%s.%s.din-%d-not", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.din-%02d-not failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + + // initialize pins + *(din->in) = 0; + *(din->in_not) = 0; + } + + // initialize digital output pins + for (i=0, dout=hal_data->douts; iindex, slave->vid, slave->pid, 0x7010, i + 1, &dout->pdo_os, &dout->pdo_bp); + + // export pins + if ((err = hal_pin_bit_newf(HAL_IN, &(dout->out), comp_id, "%s.%s.%s.dout-%d", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.dout-%02d failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_param_bit_newf(HAL_RW, &(dout->invert), comp_id, "%s.%s.%s.dout-%d-invert", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.dout-%02d-invert failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + + // initialize pins + *(dout->out) = 0; + dout->invert = 0; + } + + // initialize analog output pins + for (i=0, aout=hal_data->aouts; iindex, slave->vid, slave->pid, 0x7020 + (i << 4), 0x11, &aout->val_pdo_os, NULL); + + // export paramameters + if ((err = hal_pin_float_newf(HAL_IO, &(aout->scale), comp_id, "%s.%s.%s.aout-%d-scale", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.aout-%d-scale failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_float_newf(HAL_IO, &(aout->offset), comp_id, "%s.%s.%s.aout-%d-offset", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.aout-%d-offset failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_float_newf(HAL_IO, &(aout->min_dc), comp_id, "%s.%s.%s.aout-%d-min-dc", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.aout-%d-min-dc failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_float_newf(HAL_IO, &(aout->max_dc), comp_id, "%s.%s.%s.aout-%d-max-dc", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.aout-%d-max-dc failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_float_newf(HAL_OUT, &(aout->curr_dc), comp_id, "%s.%s.%s.aout-%d-curr-dc", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.aout-%d-curr-dc failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + + // export pins + if ((err = hal_pin_bit_newf(HAL_IN, &(aout->enable), comp_id, "%s.%s.%s.aout-%d-enable", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.aout-%d-enable failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_IN, &(aout->absmode), comp_id, "%s.%s.%s.aout-%d-absmode", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.aout-%d-absmode failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_float_newf(HAL_IN, &(aout->value), comp_id, "%s.%s.%s.aout-%d-value", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.aout-%d-value failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_s32_newf(HAL_OUT, &(aout->raw_val), comp_id, "%s.%s.%s.aout-%d-raw", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.aout-%d-raw failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + + // export UP/DOWN pins + if ((err = hal_pin_bit_newf(HAL_OUT, &(aout->pos), comp_id, "%s.%s.%s.aout-%d-pos", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.aout-%d-pos failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(aout->neg), comp_id, "%s.%s.%s.aout-%d-neg", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.aout-%d-neg failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + + // set default pin values + *(aout->scale) = 1.0; + *(aout->offset) = 0.0; + *(aout->min_dc) = -1.0; + *(aout->max_dc) = 1.0; + *(aout->curr_dc) = 0.0; + *(aout->enable) = 0; + *(aout->absmode) = 0; + *(aout->value) = 0.0; + *(aout->raw_val) = 0.0; + *(aout->pos) = 0; + *(aout->neg) = 0; + + // init other fields + aout->old_scale = *(aout->scale) + 1.0; + aout->scale_recip = 1.0; + } + + // initialize encoder pins + for (i=0, enc=hal_data->encs; iindex, slave->vid, slave->pid, 0x6060 + (i << 4), 0x02, &enc->latch_ext_valid_pdo_os, &enc->latch_ext_valid_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6060 + (i << 4), 0x03, &enc->set_count_done_pdo_os, &enc->set_count_done_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6060 + (i << 4), 0x09, &enc->ina_pdo_os, &enc->ina_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6060 + (i << 4), 0x0a, &enc->inb_pdo_os, &enc->inb_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6060 + (i << 4), 0x0c, &enc->ingate_pdo_os, &enc->ingate_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6060 + (i << 4), 0x0d, &enc->inext_pdo_os, &enc->inext_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6060 + (i << 4), 0x11, &enc->count_pdo_os, NULL); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x6060 + (i << 4), 0x12, &enc->latch_pdo_os, NULL); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x7060 + (i << 4), 0x02, &enc->ena_latch_ext_pos_pdo_os, &enc->ena_latch_ext_pos_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x7060 + (i << 4), 0x03, &enc->set_count_pdo_os, &enc->set_count_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x7060 + (i << 4), 0x04, &enc->ena_latch_ext_neg_pdo_os, &enc->ena_latch_ext_neg_pdo_bp); + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x7060 + (i << 4), 0x11, &enc->set_count_val_pdo_os, NULL); + + // export pins + if ((err = hal_pin_bit_newf(HAL_IO, &(enc->ena_latch_ext_pos), comp_id, "%s.%s.%s.enc-%d-index-ext-pos-enable", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-index-ext-pos-enable failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_IO, &(enc->ena_latch_ext_neg), comp_id, "%s.%s.%s.enc-%d-index-ext-neg-enable", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-index-ext-neg-enable failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_IN, &(enc->reset), comp_id, "%s.%s.%s.enc-%d-reset", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-reset failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(enc->ina), comp_id, "%s.%s.%s.enc-%d-ina", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-ina failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(enc->inb), comp_id, "%s.%s.%s.enc-%d-inb", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-inb failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(enc->ingate), comp_id, "%s.%s.%s.enc-%d-ingate", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-ingate failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(enc->inext), comp_id, "%s.%s.%s.enc-%d-inext", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-inext failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(enc->latch_ext_valid), comp_id, "%s.%s.%s.enc-%d-latch-ext-valid", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-latch-ext-valid failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_bit_newf(HAL_IO, &(enc->set_raw_count), comp_id, "%s.%s.%s.enc-%d-set-raw-count", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-set-raw-count failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_s32_newf(HAL_IN, &(enc->set_raw_count_val), comp_id, "%s.%s.%s.enc-%d-set-raw-count-val", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-set-raw-count-val failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_s32_newf(HAL_OUT, &(enc->raw_count), comp_id, "%s.%s.%s.enc-%d-raw-count", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-raw-count failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_s32_newf(HAL_OUT, &(enc->count), comp_id, "%s.%s.%s.enc-%d-count", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-count failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_s32_newf(HAL_OUT, &(enc->raw_latch), comp_id, "%s.%s.%s.enc-%d-raw-latch", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-raw-latch failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_float_newf(HAL_OUT, &(enc->pos), comp_id, "%s.%s.%s.enc-%d-pos", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-pos failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + if ((err = hal_pin_float_newf(HAL_IO, &(enc->pos_scale), comp_id, "%s.%s.%s.enc-%d-pos-scale", LCEC_MODULE_NAME, master->name, slave->name, i)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.enc-%d-scale failed\n", LCEC_MODULE_NAME, master->name, slave->name, i); + return err; + } + + // initialize pins + *(enc->ena_latch_ext_pos) = 0; + *(enc->ena_latch_ext_neg) = 0; + *(enc->reset) = 0; + *(enc->ina) = 0; + *(enc->inb) = 0; + *(enc->ingate) = 0; + *(enc->inext) = 0; + *(enc->latch_ext_valid) = 0; + *(enc->set_raw_count) = 0; + *(enc->set_raw_count_val) = 0; + *(enc->raw_count) = 0; + *(enc->raw_latch) = 0; + *(enc->count) = 0; + *(enc->pos) = 0; + *(enc->pos_scale) = 1.0; + + // initialize variables + enc->do_init = 1; + enc->last_count = 0; + enc->old_scale = *(enc->pos_scale) + 1.0; + enc->scale = 1.0; + } + + return 0; +} + +void lcec_em7004_read(struct lcec_slave *slave, long period) { + lcec_master_t *master = slave->master; + lcec_em7004_data_t *hal_data = (lcec_em7004_data_t *) slave->hal_data; + uint8_t *pd = master->process_data; + lcec_em7004_din_t *din; + lcec_em7004_enc_t *enc; + int i, s; + int16_t raw_count, raw_latch, raw_delta; + + // wait for slave to be operational + if (!slave->state.operational) { + hal_data->last_operational = 0; + return; + } + + // check digital inputs + for (i=0, din=hal_data->dins; ipdo_os], din->pdo_bp); + *(din->in) = s; + *(din->in_not) = !s; + } + + // read encoder data + for (i=0, enc=hal_data->encs; ipos_scale) != enc->old_scale) { + // scale value has changed, test and update it + if ((*(enc->pos_scale) < 1e-20) && (*(enc->pos_scale) > -1e-20)) { + // value too small, divide by zero is a bad thing + *(enc->pos_scale) = 1.0; + } + // save new scale to detect future changes + enc->old_scale = *(enc->pos_scale); + // we actually want the reciprocal + enc->scale = 1.0 / *(enc->pos_scale); + } + + // get bit states + *(enc->ina) = EC_READ_BIT(&pd[enc->ina_pdo_os], enc->ina_pdo_bp); + *(enc->inb) = EC_READ_BIT(&pd[enc->inb_pdo_os], enc->inb_pdo_bp); + *(enc->ingate) = EC_READ_BIT(&pd[enc->ingate_pdo_os], enc->ingate_pdo_bp); + *(enc->inext) = EC_READ_BIT(&pd[enc->inext_pdo_os], enc->inext_pdo_bp); + *(enc->latch_ext_valid) = EC_READ_BIT(&pd[enc->latch_ext_valid_pdo_os], enc->latch_ext_valid_pdo_bp); + + // read raw values + raw_count = EC_READ_S16(&pd[enc->count_pdo_os]); + raw_latch = EC_READ_S16(&pd[enc->latch_pdo_os]); + + // check for operational change of slave + if (!hal_data->last_operational) { + enc->last_count = raw_count; + } + + // check for counter set done + if (EC_READ_BIT(&pd[enc->set_count_done_pdo_os], enc->set_count_done_pdo_bp)) { + enc->last_count = raw_count; + *(enc->set_raw_count) = 0; + } + + // update raw values + if (! *(enc->set_raw_count)) { + *(enc->raw_count) = raw_count; + } + + // handle initialization + if (enc->do_init || *(enc->reset)) { + enc->do_init = 0; + enc->last_count = raw_count; + *(enc->count) = 0; + } + + // handle index + if (*(enc->latch_ext_valid)) { + *(enc->raw_latch) = raw_latch; + enc->last_count = raw_latch; + *(enc->count) = 0; + *(enc->ena_latch_ext_pos) = 0; + *(enc->ena_latch_ext_neg) = 0; + } + + // compute net counts + raw_delta = raw_count - enc->last_count; + enc->last_count = raw_count; + *(enc->count) += raw_delta; + + // scale count to make floating point position + *(enc->pos) = *(enc->count) * enc->scale; + } + + hal_data->last_operational = 1; +} + +void lcec_em7004_write(struct lcec_slave *slave, long period) { + lcec_master_t *master = slave->master; + lcec_em7004_data_t *hal_data = (lcec_em7004_data_t *) slave->hal_data; + uint8_t *pd = master->process_data; + lcec_em7004_dout_t *dout; + lcec_em7004_aout_t *aout; + lcec_em7004_enc_t *enc; + int i, s; + double tmpval, tmpdc, raw_val; + + // set digital outputs + for (i=0, dout=hal_data->douts; iout); + if (dout->invert) { + s = !s; + } + EC_WRITE_BIT(&pd[dout->pdo_os], dout->pdo_bp, s); + } + + // set analog outputs + for (i=0, aout=hal_data->aouts; imax_dc) > 1.0) { + *(aout->max_dc) = 1.0; + } + if (*(aout->min_dc) > *(aout->max_dc)) { + *(aout->min_dc) = *(aout->max_dc); + } + if (*(aout->min_dc) < -1.0) { + *(aout->min_dc) = -1.0; + } + if (*(aout->max_dc) < *(aout->min_dc)) { + *(aout->max_dc) = *(aout->min_dc); + } + + // do scale calcs only when scale changes + if (*(aout->scale) != aout->old_scale) { + // validate the new scale value + if ((*(aout->scale) < 1e-20) && (*(aout->scale) > -1e-20)) { + // value too small, divide by zero is a bad thing + *(aout->scale) = 1.0; + } + // get ready to detect future scale changes + aout->old_scale = *(aout->scale); + // we will need the reciprocal + aout->scale_recip = 1.0 / *(aout->scale); + } + + // get command + tmpval = *(aout->value); + if (*(aout->absmode) && (tmpval < 0)) { + tmpval = -tmpval; + } + + // convert value command to duty cycle + tmpdc = tmpval * aout->scale_recip + *(aout->offset); + if (tmpdc < *(aout->min_dc)) { + tmpdc = *(aout->min_dc); + } + if (tmpdc > *(aout->max_dc)) { + tmpdc = *(aout->max_dc); + } + + // set output values + if (*(aout->enable) == 0) { + raw_val = 0; + *(aout->pos) = 0; + *(aout->neg) = 0; + *(aout->curr_dc) = 0; + } else { + raw_val = (double)0x7fff * tmpdc; + if (raw_val > (double)0x7fff) { + raw_val = (double)0x7fff; + } + if (raw_val < (double)-0x7fff) { + raw_val = (double)-0x7fff; + } + *(aout->pos) = (*(aout->value) > 0); + *(aout->neg) = (*(aout->value) < 0); + *(aout->curr_dc) = tmpdc; + } + + // update value + EC_WRITE_S16(&pd[aout->val_pdo_os], (int16_t)raw_val); + *(aout->raw_val) = (int32_t)raw_val; + } + + // write encoder data + for (i=0, enc=hal_data->encs; iset_count_pdo_os], enc->set_count_pdo_bp, *(enc->set_raw_count)); + EC_WRITE_BIT(&pd[enc->ena_latch_ext_pos_pdo_os], enc->ena_latch_ext_pos_pdo_bp, *(enc->ena_latch_ext_pos)); + EC_WRITE_BIT(&pd[enc->ena_latch_ext_neg_pdo_os], enc->ena_latch_ext_neg_pdo_bp, *(enc->ena_latch_ext_neg)); + EC_WRITE_S16(&pd[enc->set_count_val_pdo_os], *(enc->set_raw_count_val)); + } +} + diff --git a/src/lcec_em7004.h b/src/lcec_em7004.h new file mode 100644 index 00000000..b2340993 --- /dev/null +++ b/src/lcec_em7004.h @@ -0,0 +1,36 @@ +// +// Copyright (C) 2014 Sascha Ittner +// +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2 of the License, or +// (at your option) any later version. +// +// This program 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 this program; if not, write to the Free Software +// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA +// +#ifndef _LCEC_EM7004_H_ +#define _LCEC_EM7004_H_ + +#include "lcec.h" + +#define LCEC_EM7004_VID LCEC_BECKHOFF_VID +#define LCEC_EM7004_PID 0x1B5C3452 + +#define LCEC_EM7004_DIN_COUNT 16 +#define LCEC_EM7004_DOUT_COUNT 16 +#define LCEC_EM7004_AOUT_COUNT 4 +#define LCEC_EM7004_ENC_COUNT 4 + +#define LCEC_EM7004_PDOS (LCEC_EM7004_DIN_COUNT + LCEC_EM7004_DOUT_COUNT + LCEC_EM7004_AOUT_COUNT + (LCEC_EM7004_ENC_COUNT * 12)) + +int lcec_em7004_init(int comp_id, struct lcec_slave *slave, ec_pdo_entry_reg_t *pdo_entry_regs); + +#endif + diff --git a/src/lcec_generic.c b/src/lcec_generic.c new file mode 100644 index 00000000..c1b72550 --- /dev/null +++ b/src/lcec_generic.c @@ -0,0 +1,305 @@ +// +// Copyright (C) 2011 Sascha Ittner +// +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2 of the License, or +// (at your option) any later version. +// +// This program 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 this program; if not, write to the Free Software +// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA +// + +#include "lcec.h" +#include "lcec_generic.h" + +void lcec_generic_read(struct lcec_slave *slave, long period); +void lcec_generic_write(struct lcec_slave *slave, long period); + +hal_s32_t lcec_generic_read_s32(uint8_t *pd, lcec_generic_pin_t *hal_data); +hal_u32_t lcec_generic_read_u32(uint8_t *pd, lcec_generic_pin_t *hal_data); +void lcec_generic_write_s32(uint8_t *pd, lcec_generic_pin_t *hal_data, hal_s32_t sval); +void lcec_generic_write_u32(uint8_t *pd, lcec_generic_pin_t *hal_data, hal_u32_t uval); + +int lcec_generic_init(int comp_id, struct lcec_slave *slave, ec_pdo_entry_reg_t *pdo_entry_regs) { + lcec_master_t *master = slave->master; + lcec_generic_pin_t *hal_data = (lcec_generic_pin_t *) slave->hal_data; + int i, j; + int err; + + // initialize callbacks + slave->proc_read = lcec_generic_read; + slave->proc_write = lcec_generic_write; + + // initialize pins + for (i=0; i < slave->pdo_entry_count; i++, hal_data++) { + // PDO mapping + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, hal_data->pdo_idx, hal_data->pdo_sidx, &hal_data->pdo_os, &hal_data->pdo_bp); + + switch (hal_data->type) { + case HAL_BIT: + if (hal_data->pdo_len == 1) { + // single bit pin + err = hal_pin_bit_newf(hal_data->dir, ((hal_bit_t **) &hal_data->pin[0]), comp_id, "%s.%s.%s.%s", LCEC_MODULE_NAME, master->name, slave->name, hal_data->name); + if (err != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.%s failed\n", LCEC_MODULE_NAME, master->name, slave->name, hal_data->name); + return err; + } + + *((hal_bit_t *) hal_data->pin[0]) = 0; + } else { + // bit pin array + for (j=0; j < LCEC_GENERIC_MAX_SUBPINS && j < hal_data->pdo_len; j++) { + err = hal_pin_bit_newf(hal_data->dir, ((hal_bit_t **) &hal_data->pin[j]), comp_id, "%s.%s.%s.%s-%d", LCEC_MODULE_NAME, master->name, slave->name, hal_data->name, j); + if (err != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.%s-%d failed\n", LCEC_MODULE_NAME, master->name, slave->name, hal_data->name, j); + return err; + } + + *((hal_bit_t *) hal_data->pin[j]) = 0; + } + } + break; + + case HAL_S32: + // check byte alignment + if (hal_data->pdo_bp != 0) { + rtapi_print_msg(RTAPI_MSG_WARN, LCEC_MSG_PFX "unable to export S32 pin %s.%s.%s.%s: process data not byte alligned!\n", LCEC_MODULE_NAME, master->name, slave->name, hal_data->name); + continue; + } + + // check data size + if (hal_data->pdo_len != 8 && hal_data->pdo_len != 16 && hal_data->pdo_len != 32) { + rtapi_print_msg(RTAPI_MSG_WARN, LCEC_MSG_PFX "unable to export S32 pin %s.%s.%s.%s: invalid process data bitlen!\n", LCEC_MODULE_NAME, master->name, slave->name, hal_data->name); + continue; + } + + // export pin + err = hal_pin_s32_newf(hal_data->dir, ((hal_s32_t **) &hal_data->pin[0]), comp_id, "%s.%s.%s.%s", LCEC_MODULE_NAME, master->name, slave->name, hal_data->name); + if (err != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.%s failed\n", LCEC_MODULE_NAME, master->name, slave->name, hal_data->name); + return err; + } + + // initialize data + *((hal_s32_t *) hal_data->pin[0]) = 0; + break; + + case HAL_U32: + // check byte alignment + if (hal_data->pdo_bp != 0) { + rtapi_print_msg(RTAPI_MSG_WARN, LCEC_MSG_PFX "unable to export U32 pin %s.%s.%s.%s: process data not byte alligned!\n", LCEC_MODULE_NAME, master->name, slave->name, hal_data->name); + continue; + } + + // check data size + if (hal_data->pdo_len != 8 && hal_data->pdo_len != 16 && hal_data->pdo_len != 32) { + rtapi_print_msg(RTAPI_MSG_WARN, LCEC_MSG_PFX "unable to export U32 pin %s.%s.%s.%s: invalid process data bitlen!\n", LCEC_MODULE_NAME, master->name, slave->name, hal_data->name); + continue; + } + + // export pin + err = hal_pin_u32_newf(hal_data->dir, ((hal_u32_t **) &hal_data->pin[0]), comp_id, "%s.%s.%s.%s", LCEC_MODULE_NAME, master->name, slave->name, hal_data->name); + if (err != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.%s failed\n", LCEC_MODULE_NAME, master->name, slave->name, hal_data->name); + return err; + } + + // initialize data + *((hal_u32_t *) hal_data->pin[0]) = 0; + break; + + case HAL_FLOAT: + // check byte alignment + if (hal_data->pdo_bp != 0) { + rtapi_print_msg(RTAPI_MSG_WARN, LCEC_MSG_PFX "unable to export FLOAT pin %s.%s.%s.%s: process data not byte alligned!\n", LCEC_MODULE_NAME, master->name, slave->name, hal_data->name); + continue; + } + + // check data size + if (hal_data->pdo_len != 8 && hal_data->pdo_len != 16 && hal_data->pdo_len != 32) { + rtapi_print_msg(RTAPI_MSG_WARN, LCEC_MSG_PFX "unable to export FLOAT pin %s.%s.%s.%s: invalid process data bitlen!\n", LCEC_MODULE_NAME, master->name, slave->name, hal_data->name); + continue; + } + + // export pin + err = hal_pin_float_newf(hal_data->dir, ((hal_float_t **) &hal_data->pin[0]), comp_id, "%s.%s.%s.%s", LCEC_MODULE_NAME, master->name, slave->name, hal_data->name); + if (err != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.%s failed\n", LCEC_MODULE_NAME, master->name, slave->name, hal_data->name); + return err; + } + + // initialize data + *((hal_float_t *) hal_data->pin[0]) = 0.0; + break; + + default: + rtapi_print_msg(RTAPI_MSG_WARN, LCEC_MSG_PFX "unsupported pin type %d!\n", hal_data->type); + } + } + + return 0; +} + +void lcec_generic_read(struct lcec_slave *slave, long period) { + lcec_master_t *master = slave->master; + lcec_generic_pin_t *hal_data = (lcec_generic_pin_t *) slave->hal_data; + uint8_t *pd = master->process_data; + int i, j, offset; + hal_float_t fval; + + // read data + for (i=0; i < slave->pdo_entry_count; i++, hal_data++) { + // skip wrong direction and uninitialized pins + if (hal_data->dir != HAL_OUT || hal_data->pin[0] == NULL) { + continue; + } + + switch (hal_data->type) { + case HAL_BIT: + offset = hal_data->pdo_os << 3 | (hal_data->pdo_bp & 0x07); + for (j=0; j < LCEC_GENERIC_MAX_SUBPINS && hal_data->pin[j] != NULL; j++, offset++) { + *((hal_bit_t *) hal_data->pin[j]) = EC_READ_BIT(&pd[offset >> 3], offset & 0x07); + } + break; + + case HAL_S32: + *((hal_s32_t *) hal_data->pin[0]) = lcec_generic_read_s32(pd, hal_data); + break; + + case HAL_U32: + *((hal_u32_t *) hal_data->pin[0]) = lcec_generic_read_u32(pd, hal_data); + break; + + case HAL_FLOAT: + if (hal_data->floatUnsigned) { + fval = lcec_generic_read_u32(pd, hal_data); + } else { + fval = lcec_generic_read_s32(pd, hal_data); + } + + fval *= hal_data->floatScale; + fval += hal_data->floatOffset; + *((hal_float_t *) hal_data->pin[0]) = fval; + break; + + default: + continue; + } + } +} + +void lcec_generic_write(struct lcec_slave *slave, long period) { + lcec_master_t *master = slave->master; + lcec_generic_pin_t *hal_data = (lcec_generic_pin_t *) slave->hal_data; + uint8_t *pd = master->process_data; + int i, j, offset; + hal_float_t fval; + + // write data + for (i=0; ipdo_entry_count; i++, hal_data++) { + // skip wrong direction and uninitialized pins + if (hal_data->dir != HAL_IN || hal_data->pin[0] == NULL) { + continue; + } + + switch (hal_data->type) { + case HAL_BIT: + offset = hal_data->pdo_os << 3 | (hal_data->pdo_bp & 0x07); + for (j=0; j < LCEC_GENERIC_MAX_SUBPINS && hal_data->pin[j] != NULL; j++, offset++) { + EC_WRITE_BIT(&pd[offset >> 3], offset & 0x07, *((hal_bit_t *) hal_data->pin[j])); + } + break; + + case HAL_S32: + lcec_generic_write_s32(pd, hal_data, *((hal_s32_t *) hal_data->pin[0])); + break; + + case HAL_U32: + lcec_generic_write_u32(pd, hal_data, *((hal_u32_t *) hal_data->pin[0])); + break; + + case HAL_FLOAT: + fval = *((hal_float_t *) hal_data->pin[0]); + fval += hal_data->floatOffset; + fval *= hal_data->floatScale; + + if (hal_data->floatUnsigned) { + lcec_generic_write_u32(pd, hal_data, (hal_u32_t) fval); + } else { + lcec_generic_write_s32(pd, hal_data, (hal_s32_t) fval); + } + break; + + default: + continue; + } + } +} + +hal_s32_t lcec_generic_read_s32(uint8_t *pd, lcec_generic_pin_t *hal_data) { + switch (hal_data->pdo_len) { + case 8: + return EC_READ_S8(&pd[hal_data->pdo_os]); + case 16: + return EC_READ_S16(&pd[hal_data->pdo_os]); + case 32: + return EC_READ_S32(&pd[hal_data->pdo_os]); + default: + return 0; + } +} + +hal_u32_t lcec_generic_read_u32(uint8_t *pd, lcec_generic_pin_t *hal_data) { + switch (hal_data->pdo_len) { + case 8: + return EC_READ_U8(&pd[hal_data->pdo_os]); + case 16: + return EC_READ_U16(&pd[hal_data->pdo_os]); + case 32: + return EC_READ_U32(&pd[hal_data->pdo_os]); + default: + return 0; + } +} + +void lcec_generic_write_s32(uint8_t *pd, lcec_generic_pin_t *hal_data, hal_s32_t sval) { + switch (hal_data->pdo_len) { + case 8: + if (sval > 0x7f) sval = 0x7f; + if (sval < -0x80) sval = -0x80; + EC_WRITE_S8(&pd[hal_data->pdo_os], sval); + return; + case 16: + if (sval > 0x7fff) sval = 0x7fff; + if (sval < -0x8000) sval = -0x8000; + EC_WRITE_S16(&pd[hal_data->pdo_os], sval); + return; + case 32: + EC_WRITE_S32(&pd[hal_data->pdo_os], sval); + return; + } +} + +void lcec_generic_write_u32(uint8_t *pd, lcec_generic_pin_t *hal_data, hal_u32_t uval) { + switch (hal_data->pdo_len) { + case 8: + if (uval > 0xff) uval = 0xff; + EC_WRITE_U8(&pd[hal_data->pdo_os], uval); + return; + case 16: + if (uval > 0xffff) uval = 0xffff; + EC_WRITE_U16(&pd[hal_data->pdo_os], uval); + return; + case 32: + EC_WRITE_U32(&pd[hal_data->pdo_os], uval); + return; + } +} + diff --git a/src/lcec_generic.h b/src/lcec_generic.h new file mode 100644 index 00000000..e3258a54 --- /dev/null +++ b/src/lcec_generic.h @@ -0,0 +1,44 @@ +// +// Copyright (C) 2011 Sascha Ittner +// +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2 of the License, or +// (at your option) any later version. +// +// This program 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 this program; if not, write to the Free Software +// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA +// +#ifndef _LCEC_GENERIC_H_ +#define _LCEC_GENERIC_H_ + +#include "lcec.h" +#include "lcec_conf.h" + +#define LCEC_GENERIC_MAX_SUBPINS 32 + +typedef struct { + char name[LCEC_CONF_STR_MAXLEN]; + hal_type_t type; + int floatUnsigned; + hal_float_t floatScale; + hal_float_t floatOffset; + hal_pin_dir_t dir; + void *pin[LCEC_GENERIC_MAX_SUBPINS]; + uint16_t pdo_idx; + uint8_t pdo_sidx; + int pdo_len; + unsigned int pdo_os; + unsigned int pdo_bp; +} lcec_generic_pin_t; + +int lcec_generic_init(int comp_id, struct lcec_slave *slave, ec_pdo_entry_reg_t *pdo_entry_regs); + +#endif + diff --git a/src/lcec_main.c b/src/lcec_main.c new file mode 100644 index 00000000..8713b4c1 --- /dev/null +++ b/src/lcec_main.c @@ -0,0 +1,1117 @@ +// +// Copyright (C) 2012 Sascha Ittner +// +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2 of the License, or +// (at your option) any later version. +// +// This program 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 this program; if not, write to the Free Software +// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA +// + +#include "lcec.h" +#include "lcec_generic.h" +#include "lcec_ek1100.h" +#include "lcec_el1xxx.h" +#include "lcec_el2xxx.h" +#include "lcec_el31x2.h" +#include "lcec_el40x1.h" +#include "lcec_el40x2.h" +#include "lcec_el41x2.h" +#include "lcec_el5101.h" +#include "lcec_el5151.h" +#include "lcec_el5152.h" +#include "lcec_el2521.h" +#include "lcec_el7342.h" +#include "lcec_el95xx.h" +#include "lcec_em7004.h" +#include "lcec_stmds5k.h" +#include "lcec_deasda.h" + +MODULE_LICENSE("GPL"); +MODULE_AUTHOR("Sascha Ittner "); +MODULE_DESCRIPTION("Driver for EtherCAT devices"); + +typedef struct lcec_typelist { + LCEC_SLAVE_TYPE_T type; + uint32_t vid; + uint32_t pid; + int pdo_entry_count; + lcec_slave_init_t proc_init; +} lcec_typelist_t; + +static const lcec_typelist_t types[] = { + // bus coupler + { lcecSlaveTypeEK1100, LCEC_EK1100_VID, LCEC_EK1100_PID, LCEC_EK1100_PDOS, NULL}, + + // digital in + { lcecSlaveTypeEL1002, LCEC_EL1xxx_VID, LCEC_EL1002_PID, LCEC_EL1002_PDOS, lcec_el1xxx_init}, + { lcecSlaveTypeEL1004, LCEC_EL1xxx_VID, LCEC_EL1004_PID, LCEC_EL1004_PDOS, lcec_el1xxx_init}, + { lcecSlaveTypeEL1008, LCEC_EL1xxx_VID, LCEC_EL1008_PID, LCEC_EL1008_PDOS, lcec_el1xxx_init}, + { lcecSlaveTypeEL1012, LCEC_EL1xxx_VID, LCEC_EL1012_PID, LCEC_EL1012_PDOS, lcec_el1xxx_init}, + { lcecSlaveTypeEL1014, LCEC_EL1xxx_VID, LCEC_EL1014_PID, LCEC_EL1014_PDOS, lcec_el1xxx_init}, + { lcecSlaveTypeEL1018, LCEC_EL1xxx_VID, LCEC_EL1018_PID, LCEC_EL1018_PDOS, lcec_el1xxx_init}, + { lcecSlaveTypeEL1024, LCEC_EL1xxx_VID, LCEC_EL1024_PID, LCEC_EL1024_PDOS, lcec_el1xxx_init}, + { lcecSlaveTypeEL1034, LCEC_EL1xxx_VID, LCEC_EL1034_PID, LCEC_EL1034_PDOS, lcec_el1xxx_init}, + { lcecSlaveTypeEL1084, LCEC_EL1xxx_VID, LCEC_EL1084_PID, LCEC_EL1084_PDOS, lcec_el1xxx_init}, + { lcecSlaveTypeEL1088, LCEC_EL1xxx_VID, LCEC_EL1088_PID, LCEC_EL1088_PDOS, lcec_el1xxx_init}, + { lcecSlaveTypeEL1094, LCEC_EL1xxx_VID, LCEC_EL1094_PID, LCEC_EL1094_PDOS, lcec_el1xxx_init}, + { lcecSlaveTypeEL1098, LCEC_EL1xxx_VID, LCEC_EL1098_PID, LCEC_EL1098_PDOS, lcec_el1xxx_init}, + { lcecSlaveTypeEL1104, LCEC_EL1xxx_VID, LCEC_EL1104_PID, LCEC_EL1104_PDOS, lcec_el1xxx_init}, + { lcecSlaveTypeEL1114, LCEC_EL1xxx_VID, LCEC_EL1114_PID, LCEC_EL1114_PDOS, lcec_el1xxx_init}, + { lcecSlaveTypeEL1124, LCEC_EL1xxx_VID, LCEC_EL1124_PID, LCEC_EL1124_PDOS, lcec_el1xxx_init}, + { lcecSlaveTypeEL1134, LCEC_EL1xxx_VID, LCEC_EL1134_PID, LCEC_EL1134_PDOS, lcec_el1xxx_init}, + { lcecSlaveTypeEL1144, LCEC_EL1xxx_VID, LCEC_EL1144_PID, LCEC_EL1144_PDOS, lcec_el1xxx_init}, + { lcecSlaveTypeEL1808, LCEC_EL1xxx_VID, LCEC_EL1808_PID, LCEC_EL1808_PDOS, lcec_el1xxx_init}, + { lcecSlaveTypeEL1809, LCEC_EL1xxx_VID, LCEC_EL1809_PID, LCEC_EL1809_PDOS, lcec_el1xxx_init}, + + // digital out + { lcecSlaveTypeEL2002, LCEC_EL2xxx_VID, LCEC_EL2002_PID, LCEC_EL2002_PDOS, lcec_el2xxx_init}, + { lcecSlaveTypeEL2004, LCEC_EL2xxx_VID, LCEC_EL2004_PID, LCEC_EL2004_PDOS, lcec_el2xxx_init}, + { lcecSlaveTypeEL2008, LCEC_EL2xxx_VID, LCEC_EL2008_PID, LCEC_EL2008_PDOS, lcec_el2xxx_init}, + { lcecSlaveTypeEL2022, LCEC_EL2xxx_VID, LCEC_EL2022_PID, LCEC_EL2022_PDOS, lcec_el2xxx_init}, + { lcecSlaveTypeEL2024, LCEC_EL2xxx_VID, LCEC_EL2024_PID, LCEC_EL2024_PDOS, lcec_el2xxx_init}, + { lcecSlaveTypeEL2032, LCEC_EL2xxx_VID, LCEC_EL2032_PID, LCEC_EL2032_PDOS, lcec_el2xxx_init}, + { lcecSlaveTypeEL2034, LCEC_EL2xxx_VID, LCEC_EL2034_PID, LCEC_EL2034_PDOS, lcec_el2xxx_init}, + { lcecSlaveTypeEL2042, LCEC_EL2xxx_VID, LCEC_EL2042_PID, LCEC_EL2042_PDOS, lcec_el2xxx_init}, + { lcecSlaveTypeEL2084, LCEC_EL2xxx_VID, LCEC_EL2084_PID, LCEC_EL2084_PDOS, lcec_el2xxx_init}, + { lcecSlaveTypeEL2088, LCEC_EL2xxx_VID, LCEC_EL2088_PID, LCEC_EL2088_PDOS, lcec_el2xxx_init}, + { lcecSlaveTypeEL2124, LCEC_EL2xxx_VID, LCEC_EL2124_PID, LCEC_EL2124_PDOS, lcec_el2xxx_init}, + { lcecSlaveTypeEL2808, LCEC_EL2xxx_VID, LCEC_EL2808_PID, LCEC_EL2808_PDOS, lcec_el2xxx_init}, + { lcecSlaveTypeEL2809, LCEC_EL2xxx_VID, LCEC_EL2809_PID, LCEC_EL2809_PDOS, lcec_el2xxx_init}, + + // analog in, 2ch, 16 bits + { lcecSlaveTypeEL3102, LCEC_EL31x2_VID, LCEC_EL3102_PID, LCEC_EL31x2_PDOS, lcec_el31x2_init}, + { lcecSlaveTypeEL3112, LCEC_EL31x2_VID, LCEC_EL3112_PID, LCEC_EL31x2_PDOS, lcec_el31x2_init}, + { lcecSlaveTypeEL3122, LCEC_EL31x2_VID, LCEC_EL3122_PID, LCEC_EL31x2_PDOS, lcec_el31x2_init}, + { lcecSlaveTypeEL3142, LCEC_EL31x2_VID, LCEC_EL3142_PID, LCEC_EL31x2_PDOS, lcec_el31x2_init}, + { lcecSlaveTypeEL3152, LCEC_EL31x2_VID, LCEC_EL3152_PID, LCEC_EL31x2_PDOS, lcec_el31x2_init}, + { lcecSlaveTypeEL3162, LCEC_EL31x2_VID, LCEC_EL3162_PID, LCEC_EL31x2_PDOS, lcec_el31x2_init}, + + // analog out, 1ch, 12 bits + { lcecSlaveTypeEL4001, LCEC_EL40x1_VID, LCEC_EL4001_PID, LCEC_EL40x1_PDOS, lcec_el40x1_init}, + { lcecSlaveTypeEL4011, LCEC_EL40x1_VID, LCEC_EL4011_PID, LCEC_EL40x1_PDOS, lcec_el40x1_init}, + { lcecSlaveTypeEL4021, LCEC_EL40x1_VID, LCEC_EL4021_PID, LCEC_EL40x1_PDOS, lcec_el40x1_init}, + { lcecSlaveTypeEL4031, LCEC_EL40x1_VID, LCEC_EL4031_PID, LCEC_EL40x1_PDOS, lcec_el40x1_init}, + + // analog out, 2ch, 12 bits + { lcecSlaveTypeEL4002, LCEC_EL40x2_VID, LCEC_EL4002_PID, LCEC_EL40x2_PDOS, lcec_el40x2_init}, + { lcecSlaveTypeEL4012, LCEC_EL40x2_VID, LCEC_EL4012_PID, LCEC_EL40x2_PDOS, lcec_el40x2_init}, + { lcecSlaveTypeEL4022, LCEC_EL40x2_VID, LCEC_EL4022_PID, LCEC_EL40x2_PDOS, lcec_el40x2_init}, + { lcecSlaveTypeEL4032, LCEC_EL40x2_VID, LCEC_EL4032_PID, LCEC_EL40x2_PDOS, lcec_el40x2_init}, + + // analog out, 2ch, 16 bits + { lcecSlaveTypeEL4102, LCEC_EL41x2_VID, LCEC_EL4102_PID, LCEC_EL41x2_PDOS, lcec_el41x2_init}, + { lcecSlaveTypeEL4112, LCEC_EL41x2_VID, LCEC_EL4112_PID, LCEC_EL41x2_PDOS, lcec_el41x2_init}, + { lcecSlaveTypeEL4122, LCEC_EL41x2_VID, LCEC_EL4122_PID, LCEC_EL41x2_PDOS, lcec_el41x2_init}, + { lcecSlaveTypeEL4132, LCEC_EL41x2_VID, LCEC_EL4132_PID, LCEC_EL41x2_PDOS, lcec_el41x2_init}, + + // encoder inputs + { lcecSlaveTypeEL5101, LCEC_EL5101_VID, LCEC_EL5101_PID, LCEC_EL5101_PDOS, lcec_el5101_init}, + { lcecSlaveTypeEL5151, LCEC_EL5151_VID, LCEC_EL5151_PID, LCEC_EL5151_PDOS, lcec_el5151_init}, + { lcecSlaveTypeEL5152, LCEC_EL5152_VID, LCEC_EL5152_PID, LCEC_EL5152_PDOS, lcec_el5152_init}, + + // pulse train (stepper) output + { lcecSlaveTypeEL2521, LCEC_EL2521_VID, LCEC_EL2521_PID, LCEC_EL2521_PDOS, lcec_el2521_init}, + + // dc servo + { lcecSlaveTypeEL7342, LCEC_EL7342_VID, LCEC_EL7342_PID, LCEC_EL7342_PDOS, lcec_el7342_init}, + + // power supply + { lcecSlaveTypeEL9505, LCEC_EL95xx_VID, LCEC_EL9505_PID, LCEC_EL95xx_PDOS, lcec_el95xx_init}, + { lcecSlaveTypeEL9508, LCEC_EL95xx_VID, LCEC_EL9508_PID, LCEC_EL95xx_PDOS, lcec_el95xx_init}, + { lcecSlaveTypeEL9510, LCEC_EL95xx_VID, LCEC_EL9510_PID, LCEC_EL95xx_PDOS, lcec_el95xx_init}, + { lcecSlaveTypeEL9512, LCEC_EL95xx_VID, LCEC_EL9512_PID, LCEC_EL95xx_PDOS, lcec_el95xx_init}, + { lcecSlaveTypeEL9515, LCEC_EL95xx_VID, LCEC_EL9515_PID, LCEC_EL95xx_PDOS, lcec_el95xx_init}, + + // multi axis interface + { lcecSlaveTypeEM7004, LCEC_EM7004_VID, LCEC_EM7004_PID, LCEC_EM7004_PDOS, lcec_em7004_init}, + + // stoeber MDS5000 series + { lcecSlaveTypeStMDS5k, LCEC_STMDS5K_VID, LCEC_STMDS5K_PID, LCEC_STMDS5K_PDOS, lcec_stmds5k_init}, + + // Delta ASDA series + { lcecSlaveTypeDeASDA, LCEC_DEASDA_VID, LCEC_DEASDA_PID, LCEC_DEASDA_PDOS, lcec_deasda_init}, + + { lcecSlaveTypeInvalid } +}; + +static lcec_master_t *first_master = NULL; +static lcec_master_t *last_master = NULL; +static int comp_id = -1; + +static lcec_master_data_t *global_hal_data; +static ec_master_state_t global_ms; + +int lcec_parse_config(void); +void lcec_clear_config(void); + +void lcec_request_lock(void *data); +void lcec_release_lock(void *data); + +lcec_master_data_t *lcec_init_master_hal(const char *pfx); +lcec_slave_state_t *lcec_init_slave_state_hal(char *master_name, char *slave_name); +void lcec_update_master_hal(lcec_master_data_t *hal_data, ec_master_state_t *ms); +void lcec_update_slave_state_hal(lcec_slave_state_t *hal_data, ec_slave_config_state_t *ss); + +void lcec_read_all(void *arg, long period); +void lcec_write_all(void *arg, long period); +void lcec_read_master(void *arg, long period); +void lcec_write_master(void *arg, long period); + +int rtapi_app_main(void) { + int slave_count; + lcec_master_t *master; + lcec_slave_t *slave; + char name[HAL_NAME_LEN + 1]; + ec_pdo_entry_reg_t *pdo_entry_regs; + lcec_slave_sdoconf_t *sdo_config; + struct timeval tv; + + // connect to the HAL + if ((comp_id = hal_init (LCEC_MODULE_NAME)) < 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "hal_init() failed\n"); + goto fail0; + } + + // parse configuration + if ((slave_count = lcec_parse_config()) < 0) { + goto fail1; + } + + // init global hal data + if ((global_hal_data = lcec_init_master_hal(LCEC_MODULE_NAME)) == NULL) { + goto fail2; + } + + // initialize masters + for (master = first_master; master != NULL; master = master->next) { + // request ethercat master + if (!(master->master = ecrt_request_master(master->index))) { + rtapi_print_msg (RTAPI_MSG_ERR, LCEC_MSG_PFX "requesting master %s (index %d) failed\n", master->name, master->index); + goto fail2; + } + +#ifdef __KERNEL__ + // register callbacks + ecrt_master_callbacks(master->master, lcec_request_lock, lcec_release_lock, master); +#endif + + // create domain + if (!(master->domain = ecrt_master_create_domain(master->master))) { + rtapi_print_msg (RTAPI_MSG_ERR, LCEC_MSG_PFX "master %s domain creation failed\n", master->name); + goto fail2; + } + + // initialize slaves + pdo_entry_regs = master->pdo_entry_regs; + for (slave = master->first_slave; slave != NULL; slave = slave->next) { + // read slave config + if (!(slave->config = ecrt_master_slave_config(master->master, 0, slave->index, slave->vid, slave->pid))) { + rtapi_print_msg (RTAPI_MSG_ERR, LCEC_MSG_PFX "fail to read slave %s.%s configuration\n", master->name, slave->name); + goto fail2; + } + + // initialize sdos + if (slave->sdo_config != NULL) { + for (sdo_config = slave->sdo_config; sdo_config->index != 0xffff; sdo_config = (lcec_slave_sdoconf_t *) &sdo_config->data[sdo_config->length]) { + if (sdo_config->subindex == LCEC_CONF_SDO_COMPLETE_SUBIDX) { + if (ecrt_slave_config_complete_sdo(slave->config, sdo_config->index, &sdo_config->data[0], sdo_config->length) != 0) { + rtapi_print_msg (RTAPI_MSG_ERR, LCEC_MSG_PFX "fail to configure slave %s.%s sdo %04x (complete)\n", master->name, slave->name, sdo_config->index); + } + } else { + if (ecrt_slave_config_sdo(slave->config, sdo_config->index, sdo_config->subindex, &sdo_config->data[0], sdo_config->length) != 0) { + rtapi_print_msg (RTAPI_MSG_ERR, LCEC_MSG_PFX "fail to configure slave %s.%s sdo %04x:%02x\n", master->name, slave->name, sdo_config->index, sdo_config->subindex); + } + } + } + } + + // setup pdos + if (slave->proc_init != NULL) { + if ((slave->proc_init(comp_id, slave, pdo_entry_regs)) != 0) { + goto fail2; + } + } + pdo_entry_regs += slave->pdo_entry_count; + + // configure dc for this slave + if (slave->dc_conf != NULL) { + ecrt_slave_config_dc(slave->config, slave->dc_conf->assignActivate, + slave->dc_conf->sync0Cycle, slave->dc_conf->sync0Shift, + slave->dc_conf->sync1Cycle, slave->dc_conf->sync1Shift); + rtapi_print_msg (RTAPI_MSG_DBG, LCEC_MSG_PFX "configuring DC for slave %s.%s: assignActivate=x%x sync0Cycle=%d sync0Shift=%d sync1Cycle=%d sync1Shift=%d\n", + master->name, slave->name, slave->dc_conf->assignActivate, + slave->dc_conf->sync0Cycle, slave->dc_conf->sync0Shift, + slave->dc_conf->sync1Cycle, slave->dc_conf->sync1Shift); + } + + // Configure the slave's watchdog times. + if (slave->wd_conf != NULL) { + ecrt_slave_config_watchdog(slave->config, slave->wd_conf->divider, slave->wd_conf->intervals); + } + + // configure slave + if (slave->sync_info != NULL) { + if (ecrt_slave_config_pdos(slave->config, EC_END, slave->sync_info)) { + rtapi_print_msg (RTAPI_MSG_ERR, LCEC_MSG_PFX "fail to configure slave %s.%s\n", master->name, slave->name); + goto fail2; + } + } + + // export state pins + if ((slave->hal_state_data = lcec_init_slave_state_hal(master->name, slave->name)) == NULL) { + goto fail2; + } + } + + // terminate POD entries + pdo_entry_regs->index = 0; + + // register PDO entries + if (ecrt_domain_reg_pdo_entry_list(master->domain, master->pdo_entry_regs)) { + rtapi_print_msg (RTAPI_MSG_ERR, LCEC_MSG_PFX "master %s PDO entry registration failed\n", master->name); + goto fail2; + } + + // initialize application time + RTAPI_GETTIMEOFDAY(&tv); + master->app_time = EC_TIMEVAL2NANO(tv); + + // activating master + if (ecrt_master_activate(master->master)) { + rtapi_print_msg (RTAPI_MSG_ERR, LCEC_MSG_PFX "failed to activate master %s\n", master->name); + goto fail2; + } + + // Get internal process data for domain + master->process_data = ecrt_domain_data(master->domain); + master->process_data_len = ecrt_domain_size(master->domain); + + // init hal data + rtapi_snprintf(name, HAL_NAME_LEN, "%s.%s", LCEC_MODULE_NAME, master->name); + if ((master->hal_data = lcec_init_master_hal(name)) == NULL) { + goto fail2; + } + + // export read function + rtapi_snprintf(name, HAL_NAME_LEN, "%s.%s.read", LCEC_MODULE_NAME, master->name); + if (hal_export_funct(name, lcec_read_master, master, 0, 0, comp_id) != 0) { + rtapi_print_msg (RTAPI_MSG_ERR, LCEC_MSG_PFX "master %s read funct export failed\n", master->name); + goto fail2; + } + // export write function + rtapi_snprintf(name, HAL_NAME_LEN, "%s.%s.write", LCEC_MODULE_NAME, master->name); + if (hal_export_funct(name, lcec_write_master, master, 0, 0, comp_id) != 0) { + rtapi_print_msg (RTAPI_MSG_ERR, LCEC_MSG_PFX "master %s write funct export failed\n", master->name); + goto fail2; + } + } + + // export read-all function + rtapi_snprintf(name, HAL_NAME_LEN, "%s.read-all", LCEC_MODULE_NAME); + if (hal_export_funct(name, lcec_read_all, NULL, 0, 0, comp_id) != 0) { + rtapi_print_msg (RTAPI_MSG_ERR, LCEC_MSG_PFX "read-all funct export failed\n"); + goto fail2; + } + // export write-all function + rtapi_snprintf(name, HAL_NAME_LEN, "%s.write-all", LCEC_MODULE_NAME); + if (hal_export_funct(name, lcec_write_all, NULL, 0, 0, comp_id) != 0) { + rtapi_print_msg (RTAPI_MSG_ERR, LCEC_MSG_PFX "write-all funct export failed\n"); + goto fail2; + } + + rtapi_print_msg(RTAPI_MSG_INFO, LCEC_MSG_PFX "installed driver for %d slaves\n", slave_count); + hal_ready (comp_id); + return 0; + +fail2: + lcec_clear_config(); +fail1: + hal_exit(comp_id); +fail0: + return -EINVAL; +} + +void rtapi_app_exit(void) { + lcec_master_t *master; + + // deactivate all masters + for (master = first_master; master != NULL; master = master->next) { + ecrt_master_deactivate(master->master); + } + + lcec_clear_config(); + hal_exit(comp_id); +} + +int lcec_parse_config(void) { + int shmem_id; + void *shmem_ptr; + LCEC_CONF_HEADER_T *header; + size_t length; + void *conf; + int slave_count; + const lcec_typelist_t *type; + lcec_master_t *master; + lcec_slave_t *slave; + lcec_slave_dc_t *dc; + lcec_slave_watchdog_t *wd; + ec_pdo_entry_reg_t *pdo_entry_regs; + LCEC_CONF_TYPE_T conf_type; + LCEC_CONF_MASTER_T *master_conf; + LCEC_CONF_SLAVE_T *slave_conf; + LCEC_CONF_DC_T *dc_conf; + LCEC_CONF_WATCHDOG_T *wd_conf; + LCEC_CONF_SYNCMANAGER_T *sm_conf; + LCEC_CONF_PDO_T *pdo_conf; + LCEC_CONF_PDOENTRY_T *pe_conf; + LCEC_CONF_SDOCONF_T *sdo_conf; + ec_pdo_entry_info_t *generic_pdo_entries; + ec_pdo_info_t *generic_pdos; + ec_sync_info_t *generic_sync_managers; + lcec_generic_pin_t *generic_hal_data; + hal_pin_dir_t generic_hal_dir; + lcec_slave_sdoconf_t *sdo_config; + + // initialize list + first_master = NULL; + last_master = NULL; + + // try to get config header + shmem_id = rtapi_shmem_new(LCEC_CONF_SHMEM_KEY, comp_id, sizeof(LCEC_CONF_HEADER_T)); + if (shmem_id < 0) { + rtapi_print_msg (RTAPI_MSG_ERR, LCEC_MSG_PFX "couldn't allocate user/RT shared memory\n"); + goto fail0; + } + if (rtapi_shmem_getptr(shmem_id, &shmem_ptr) < 0 ) { + rtapi_print_msg (RTAPI_MSG_ERR, LCEC_MSG_PFX "couldn't map user/RT shared memory\n"); + goto fail1; + } + + // check magic, get length and close shmem + header = shmem_ptr; + if (header->magic != LCEC_CONF_SHMEM_MAGIC) { + rtapi_print_msg (RTAPI_MSG_ERR, LCEC_MSG_PFX "lcec_conf is not loaded\n"); + goto fail1; + } + length = header->length; + rtapi_shmem_delete(shmem_id, comp_id); + + // reopen shmem with proper size + shmem_id = rtapi_shmem_new(LCEC_CONF_SHMEM_KEY, comp_id, sizeof(LCEC_CONF_HEADER_T) + length); + if (shmem_id < 0) { + rtapi_print_msg (RTAPI_MSG_ERR, LCEC_MSG_PFX "couldn't allocate user/RT shared memory\n"); + goto fail0; + } + if (rtapi_shmem_getptr(shmem_id, &shmem_ptr) < 0 ) { + rtapi_print_msg (RTAPI_MSG_ERR, LCEC_MSG_PFX "couldn't map user/RT shared memory\n"); + goto fail1; + } + + // get pointer to config + conf = shmem_ptr + sizeof(LCEC_CONF_HEADER_T); + + // process config items + slave_count = 0; + master = NULL; + slave = NULL; + generic_pdo_entries = NULL; + generic_pdos = NULL; + generic_sync_managers = NULL; + generic_hal_data = NULL; + generic_hal_dir = 0; + sdo_config = NULL; + while((conf_type = ((LCEC_CONF_NULL_T *)conf)->confType) != lcecConfTypeNone) { + // get type + switch (conf_type) { + case lcecConfTypeMaster: + // get config token + master_conf = (LCEC_CONF_MASTER_T *)conf; + conf += sizeof(LCEC_CONF_MASTER_T); + + // alloc master memory + master = kzalloc(sizeof(lcec_master_t), GFP_KERNEL); + if (master == NULL) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "Unable to allocate master %d structure memory\n", master_conf->index); + goto fail2; + } + + // initialize master + master->index = master_conf->index; + strncpy(master->name, master_conf->name, LCEC_CONF_STR_MAXLEN); + master->name[LCEC_CONF_STR_MAXLEN - 1] = 0; + master->mutex = 0; + master->app_time = 0; + master->app_time_period = master_conf->appTimePeriod; + master->sync_ref_cnt = 0; + master->sync_ref_cycles = master_conf->refClockSyncCycles; + + // add master to list + LCEC_LIST_APPEND(first_master, last_master, master); + break; + + case lcecConfTypeSlave: + // get config token + slave_conf = (LCEC_CONF_SLAVE_T *)conf; + conf += sizeof(LCEC_CONF_SLAVE_T); + + // check for master + if (master == NULL) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "Master node for slave missing\n"); + goto fail2; + } + + // check for valid slave type + if (slave_conf->type == lcecSlaveTypeGeneric) { + type = NULL; + } else { + for (type = types; type->type != slave_conf->type && type->type != lcecSlaveTypeInvalid; type++); + if (type->type == lcecSlaveTypeInvalid) { + rtapi_print_msg(RTAPI_MSG_WARN, LCEC_MSG_PFX "Invalid slave type %d\n", slave_conf->type); + continue; + } + } + + // create new slave + slave = kzalloc(sizeof(lcec_slave_t), GFP_KERNEL); + if (slave == NULL) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "Unable to allocate slave %s.%s structure memory\n", master->name, slave_conf->name); + goto fail2; + } + + // initialize slave + generic_pdo_entries = NULL; + generic_pdos = NULL; + generic_sync_managers = NULL; + generic_hal_data = NULL; + generic_hal_dir = 0; + sdo_config = NULL; + + slave->index = slave_conf->index; + strncpy(slave->name, slave_conf->name, LCEC_CONF_STR_MAXLEN); + slave->name[LCEC_CONF_STR_MAXLEN - 1] = 0; + slave->master = master; + + // add slave to list + LCEC_LIST_APPEND(master->first_slave, master->last_slave, slave); + + if (type != NULL) { + // normal slave + slave->vid = type->vid; + slave->pid = type->pid; + slave->pdo_entry_count = type->pdo_entry_count; + slave->proc_init = type->proc_init; + } else { + // generic slave + slave->vid = slave_conf->vid; + slave->pid = slave_conf->pid; + slave->pdo_entry_count = slave_conf->pdoMappingCount; + slave->proc_init = lcec_generic_init; + + // alloc hal memory + if ((generic_hal_data = hal_malloc(sizeof(lcec_generic_pin_t) * slave_conf->pdoMappingCount)) == NULL) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "hal_malloc() for slave %s.%s failed\n", master->name, slave_conf->name); + goto fail2; + } + memset(generic_hal_data, 0, sizeof(lcec_generic_pin_t) * slave_conf->pdoMappingCount); + + // alloc pdo entry memory + generic_pdo_entries = kzalloc(sizeof(ec_pdo_entry_info_t) * slave_conf->pdoEntryCount, GFP_KERNEL); + if (generic_pdo_entries == NULL) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "Unable to allocate slave %s.%s generic pdo entry memory\n", master->name, slave_conf->name); + goto fail2; + } + + // alloc pdo memory + generic_pdos = kzalloc(sizeof(ec_pdo_info_t) * slave_conf->pdoCount, GFP_KERNEL); + if (generic_pdos == NULL) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "Unable to allocate slave %s.%s generic pdo memory\n", master->name, slave_conf->name); + goto fail2; + } + + // alloc sync manager memory + generic_sync_managers = kzalloc(sizeof(ec_sync_info_t) * (slave_conf->syncManagerCount + 1), GFP_KERNEL); + if (generic_sync_managers == NULL) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "Unable to allocate slave %s.%s generic sync manager memory\n", master->name, slave_conf->name); + goto fail2; + } + generic_sync_managers->index = 0xff; + } + + // alloc sdo config memory + if (slave_conf->sdoConfigLength > 0) { + sdo_config = kzalloc(slave_conf->sdoConfigLength + sizeof(lcec_slave_sdoconf_t), GFP_KERNEL); + if (sdo_config == NULL) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "Unable to allocate slave %s.%s generic pdo entry memory\n", master->name, slave_conf->name); + goto fail2; + } + } + + slave->hal_data = generic_hal_data; + slave->generic_pdo_entries = generic_pdo_entries; + slave->generic_pdos = generic_pdos; + slave->generic_sync_managers = generic_sync_managers; + if (slave_conf->configPdos) { + slave->sync_info = generic_sync_managers; + } + slave->sdo_config = sdo_config; + slave->dc_conf = NULL; + slave->wd_conf = NULL; + + // update master's POD entry count + master->pdo_entry_count += slave->pdo_entry_count; + + // update slave count + slave_count++; + break; + + case lcecConfTypeDcConf: + // get config token + dc_conf = (LCEC_CONF_DC_T *)conf; + conf += sizeof(LCEC_CONF_DC_T); + + // check for slave + if (slave == NULL) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "Slave node for dc config missing\n"); + goto fail2; + } + + // check for double dc config + if (slave->dc_conf != NULL) { + rtapi_print_msg(RTAPI_MSG_WARN, LCEC_MSG_PFX "Double dc config for slave %s.%s\n", master->name, slave->name); + continue; + } + + // create new dc config + dc = kzalloc(sizeof(lcec_slave_dc_t), GFP_KERNEL); + if (dc == NULL) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "Unable to allocate slave %s.%s dc config memory\n", master->name, slave->name); + goto fail2; + } + + // initialize dc conf + dc->assignActivate = dc_conf->assignActivate; + dc->sync0Cycle = dc_conf->sync0Cycle; + dc->sync0Shift = dc_conf->sync0Shift; + dc->sync1Cycle = dc_conf->sync1Cycle; + dc->sync1Shift = dc_conf->sync1Shift; + + // add to slave + slave->dc_conf = dc; + break; + + case lcecConfTypeWatchdog: + // get config token + wd_conf = (LCEC_CONF_WATCHDOG_T *)conf; + conf += sizeof(LCEC_CONF_WATCHDOG_T); + + // check for slave + if (slave == NULL) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "Slave node for watchdog config missing\n"); + goto fail2; + } + + // check for double wd config + if (slave->wd_conf != NULL) { + rtapi_print_msg(RTAPI_MSG_WARN, LCEC_MSG_PFX "Double watchdog config for slave %s.%s\n", master->name, slave->name); + continue; + } + + // create new wd config + wd = kzalloc(sizeof(lcec_slave_watchdog_t), GFP_KERNEL); + if (wd == NULL) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "Unable to allocate slave %s.%s watchdog config memory\n", master->name, slave->name); + goto fail2; + } + + // initialize wd conf + wd->divider = wd_conf->divider; + wd->intervals = wd_conf->intervals; + + // add to slave + slave->wd_conf = wd; + break; + + case lcecConfTypeSyncManager: + // get config token + sm_conf = (LCEC_CONF_SYNCMANAGER_T *)conf; + conf += sizeof(LCEC_CONF_SYNCMANAGER_T); + + // check for syncmanager + if (generic_sync_managers == NULL) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "Sync manager for generic device missing\n"); + goto fail2; + } + + // check for pdos + if (generic_pdos == NULL) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "PDOs for generic device missing\n"); + goto fail2; + } + + // initialize sync manager + generic_sync_managers->index = sm_conf->index; + generic_sync_managers->dir = sm_conf->dir; + generic_sync_managers->n_pdos = sm_conf->pdoCount; + generic_sync_managers->pdos = sm_conf->pdoCount == 0 ? NULL : generic_pdos; + + // get hal direction + switch (sm_conf->dir) { + case EC_DIR_INPUT: + generic_hal_dir = HAL_OUT; + break; + case EC_DIR_OUTPUT: + generic_hal_dir = HAL_IN; + break; + default: + generic_hal_dir = 0; + } + + // next syncmanager + generic_sync_managers++; + generic_sync_managers->index = 0xff; + break; + + case lcecConfTypePdo: + // get config token + pdo_conf = (LCEC_CONF_PDO_T *)conf; + conf += sizeof(LCEC_CONF_PDO_T); + + // check for pdos + if (generic_pdos == NULL) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "PDOs for generic device missing\n"); + goto fail2; + } + + // check for pdos entries + if (generic_pdo_entries == NULL) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "PDO entries for generic device missing\n"); + goto fail2; + } + + // initialize pdo + generic_pdos->index = pdo_conf->index; + generic_pdos->n_entries = pdo_conf->pdoEntryCount; + generic_pdos->entries = pdo_conf->pdoEntryCount == 0 ? NULL : generic_pdo_entries; + + // next pdo + generic_pdos++; + break; + + case lcecConfTypePdoEntry: + // get config token + pe_conf = (LCEC_CONF_PDOENTRY_T *)conf; + conf += sizeof(LCEC_CONF_PDOENTRY_T); + + // check for pdos entries + if (generic_pdo_entries == NULL) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "PDO entries for generic device missing\n"); + goto fail2; + } + + // check for hal data + if (generic_hal_data == NULL) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "HAL data for generic device missing\n"); + goto fail2; + } + + // check for hal dir + if (generic_hal_dir == 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "HAL direction for generic device missing\n"); + goto fail2; + } + + // initialize pdo entry + generic_pdo_entries->index = pe_conf->index; + generic_pdo_entries->subindex = pe_conf->subindex; + generic_pdo_entries->bit_length = pe_conf->bitLength; + + // initialize hal data + if (pe_conf->halPin[0] != 0) { + strncpy(generic_hal_data->name, pe_conf->halPin, LCEC_CONF_STR_MAXLEN); + generic_hal_data->name[LCEC_CONF_STR_MAXLEN - 1] = 0; + generic_hal_data->type = pe_conf->halType; + generic_hal_data->floatUnsigned = pe_conf->floatUnsigned; + generic_hal_data->floatScale = pe_conf->floatScale; + generic_hal_data->floatOffset = pe_conf->floatOffset; + generic_hal_data->dir = generic_hal_dir; + generic_hal_data->pdo_idx = pe_conf->index; + generic_hal_data->pdo_sidx = pe_conf->subindex; + generic_hal_data->pdo_len = pe_conf->bitLength; + generic_hal_data++; + } + + // next pdo entry + generic_pdo_entries++; + break; + + case lcecConfTypeSdoConfig: + // get config token + sdo_conf = (LCEC_CONF_SDOCONF_T *)conf; + conf += sizeof(LCEC_CONF_SDOCONF_T) + sdo_conf->length; + + // copy attributes + sdo_config->index = sdo_conf->index; + sdo_config->subindex = sdo_conf->subindex; + sdo_config->length = sdo_conf->length; + + // copy data + memcpy(sdo_config->data, sdo_conf->data, sdo_config->length); + + sdo_config = (lcec_slave_sdoconf_t *) &sdo_config->data[sdo_config->length]; + sdo_config->index = 0xffff; + break; + + default: + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "Unknown config item type\n"); + goto fail2; + } + } + + // close shmem + rtapi_shmem_delete(shmem_id, comp_id); + + // allocate PDO entity memory + for (master = first_master; master != NULL; master = master->next) { + pdo_entry_regs = kzalloc(sizeof(ec_pdo_entry_reg_t) * (master->pdo_entry_count + 1), GFP_KERNEL); + if (pdo_entry_regs == NULL) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "Unable to allocate master %s PDO entry memory\n", master->name); + goto fail2; + } + master->pdo_entry_regs = pdo_entry_regs; + } + + return slave_count; + +fail2: + lcec_clear_config(); +fail1: + rtapi_shmem_delete(shmem_id, comp_id); +fail0: + return -1; +} + +void lcec_clear_config(void) { + lcec_master_t *master, *prev_master; + lcec_slave_t *slave, *prev_slave; + + // iterate all masters + master = last_master; + while (master != NULL) { + prev_master = master->prev; + + // iterate all masters + slave = master->last_slave; + while (slave != NULL) { + prev_slave = slave->prev; + + // cleanup slave + if (slave->proc_cleanup != NULL) { + slave->proc_cleanup(slave); + } + + // free slave + if (slave->sdo_config != NULL) { + kfree(slave->sdo_config); + } + if (slave->generic_pdo_entries != NULL) { + kfree(slave->generic_pdo_entries); + } + if (slave->generic_pdos != NULL) { + kfree(slave->generic_pdos); + } + if (slave->generic_sync_managers != NULL) { + kfree(slave->generic_sync_managers); + } + if (slave->dc_conf != NULL) { + kfree(slave->dc_conf); + } + if (slave->wd_conf != NULL) { + kfree(slave->wd_conf); + } + kfree(slave); + slave = prev_slave; + } + + // release master + if (master->master) { + ecrt_release_master(master->master); + } + + // free PDO entry memory + if (master->pdo_entry_regs != NULL) { + kfree(master->pdo_entry_regs); + } + + // free master + kfree(master); + master = prev_master; + } +} + +void lcec_request_lock(void *data) { + lcec_master_t *master = (lcec_master_t *) data; + rtapi_mutex_get(&master->mutex); +} + +void lcec_release_lock(void *data) { + lcec_master_t *master = (lcec_master_t *) data; + rtapi_mutex_give(&master->mutex); +} + +lcec_master_data_t *lcec_init_master_hal(const char *pfx) { + lcec_master_data_t *hal_data; + + // alloc hal data + if ((hal_data = hal_malloc(sizeof(lcec_master_data_t))) == NULL) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "hal_malloc() for %s failed\n", pfx); + return NULL; + } + memset(hal_data, 0, sizeof(lcec_master_data_t)); + + // export pins + if (hal_pin_u32_newf(HAL_OUT, &(hal_data->slaves_responding), comp_id, "%s.slaves-responding", pfx) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.slaves-responding failed\n", pfx); + return NULL; + } + if (hal_pin_bit_newf(HAL_OUT, &(hal_data->state_init), comp_id, "%s.state-init", pfx) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.state-init failed\n", pfx); + return NULL; + } + if (hal_pin_bit_newf(HAL_OUT, &(hal_data->state_preop), comp_id, "%s.state-preop", pfx) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.state-preop failed\n", pfx); + return NULL; + } + if (hal_pin_bit_newf(HAL_OUT, &(hal_data->state_safeop), comp_id, "%s.state-safeop", pfx) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.state-safeop failed\n", pfx); + return NULL; + } + if (hal_pin_bit_newf(HAL_OUT, &(hal_data->state_op), comp_id, "%s.state-op", pfx) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.state-op failed\n", pfx); + return NULL; + } + if (hal_pin_bit_newf(HAL_OUT, &(hal_data->link_up), comp_id, "%s.link-up", pfx) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.link-up failed\n", pfx); + return NULL; + } + if (hal_pin_bit_newf(HAL_OUT, &(hal_data->all_op), comp_id, "%s.all-op", pfx) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.all-op failed\n", pfx); + return NULL; + } + + // initialize pins + *(hal_data->slaves_responding) = 0; + *(hal_data->state_init) = 0; + *(hal_data->state_preop) = 0; + *(hal_data->state_safeop) = 0; + *(hal_data->state_op) = 0; + *(hal_data->link_up) = 0; + *(hal_data->all_op) = 0; + + return hal_data; +} + +lcec_slave_state_t *lcec_init_slave_state_hal(char *master_name, char *slave_name) { + lcec_slave_state_t *hal_data; + + // alloc hal data + if ((hal_data = hal_malloc(sizeof(lcec_slave_state_t))) == NULL) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "hal_malloc() for %s.%s.%s failed\n", LCEC_MODULE_NAME, master_name, slave_name); + return NULL; + } + memset(hal_data, 0, sizeof(lcec_master_data_t)); + + // export pins + if (hal_pin_bit_newf(HAL_OUT, &(hal_data->online), comp_id, "%s.%s.%s.slave-online", LCEC_MODULE_NAME, master_name, slave_name) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.slaves-online failed\n", LCEC_MODULE_NAME, master_name, slave_name); + return NULL; + } + if (hal_pin_bit_newf(HAL_OUT, &(hal_data->operational), comp_id, "%s.%s.%s.slave-oper", LCEC_MODULE_NAME, master_name, slave_name) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.slaves-oper failed\n", LCEC_MODULE_NAME, master_name, slave_name); + return NULL; + } + if (hal_pin_bit_newf(HAL_OUT, &(hal_data->state_init), comp_id, "%s.%s.%s.slave-state-init", LCEC_MODULE_NAME, master_name, slave_name) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.state-state-init failed\n", LCEC_MODULE_NAME, master_name, slave_name); + return NULL; + } + if (hal_pin_bit_newf(HAL_OUT, &(hal_data->state_preop), comp_id, "%s.%s.%s.slave-state-preop", LCEC_MODULE_NAME, master_name, slave_name) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.state-state-preop failed\n", LCEC_MODULE_NAME, master_name, slave_name); + return NULL; + } + if (hal_pin_bit_newf(HAL_OUT, &(hal_data->state_safeop), comp_id, "%s.%s.%s.slave-state-safeop", LCEC_MODULE_NAME, master_name, slave_name) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.state-state-safeop failed\n", LCEC_MODULE_NAME, master_name, slave_name); + return NULL; + } + if (hal_pin_bit_newf(HAL_OUT, &(hal_data->state_op), comp_id, "%s.%s.%s.slave-state-op", LCEC_MODULE_NAME, master_name, slave_name) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.state-state-op failed\n", LCEC_MODULE_NAME, master_name, slave_name); + return NULL; + } + + // initialize pins + *(hal_data->online) = 0; + *(hal_data->operational) = 0; + *(hal_data->state_init) = 0; + *(hal_data->state_preop) = 0; + *(hal_data->state_safeop) = 0; + *(hal_data->state_op) = 0; + + return hal_data; +} + +void lcec_update_master_hal(lcec_master_data_t *hal_data, ec_master_state_t *ms) { + *(hal_data->slaves_responding) = ms->slaves_responding; + *(hal_data->state_init) = (ms->al_states & 0x01) != 0; + *(hal_data->state_preop) = (ms->al_states & 0x02) != 0; + *(hal_data->state_safeop) = (ms->al_states & 0x04) != 0; + *(hal_data->state_op) = (ms->al_states & 0x08) != 0; + *(hal_data->link_up) = ms->link_up; + *(hal_data->all_op) = (ms->al_states == 0x08); +} + +void lcec_update_slave_state_hal(lcec_slave_state_t *hal_data, ec_slave_config_state_t *ss) { + *(hal_data->online) = ss->online; + *(hal_data->operational) = ss->operational; + *(hal_data->state_init) = (ss->al_state & 0x01) != 0; + *(hal_data->state_preop) = (ss->al_state & 0x02) != 0; + *(hal_data->state_safeop) = (ss->al_state & 0x04) != 0; + *(hal_data->state_op) = (ss->al_state & 0x08) != 0; +} + +void lcec_read_all(void *arg, long period) { + lcec_master_t *master; + + // initialize global state + global_ms.slaves_responding = 0; + global_ms.al_states = 0; + global_ms.link_up = (first_master != NULL); + + // process slaves + for (master = first_master; master != NULL; master = master->next) { + lcec_read_master(master, period); + } + + // update global state pins + lcec_update_master_hal(global_hal_data, &global_ms); +} + +void lcec_write_all(void *arg, long period) { + lcec_master_t *master; + + // process slaves + for (master = first_master; master != NULL; master = master->next) { + lcec_write_master(master, period); + } +} + +void lcec_read_master(void *arg, long period) { + lcec_master_t *master = (lcec_master_t *) arg; + lcec_slave_t *slave; + ec_master_state_t ms; + + // receive process data & master state + rtapi_mutex_get(&master->mutex); + ecrt_master_receive(master->master); + ecrt_domain_process(master->domain); + ecrt_master_state(master->master, &ms); + rtapi_mutex_give(&master->mutex); + + // update state pins + lcec_update_master_hal(master->hal_data, &ms); + + // update global state + global_ms.slaves_responding += ms.slaves_responding; + global_ms.al_states |= ms.al_states; + global_ms.link_up = global_ms.link_up && ms.link_up; + + // process slaves + for (slave = master->first_slave; slave != NULL; slave = slave->next) { + // get slaves state + rtapi_mutex_get(&master->mutex); + ecrt_slave_config_state(slave->config, &slave->state); + rtapi_mutex_give(&master->mutex); + lcec_update_slave_state_hal(slave->hal_state_data, &slave->state); + + // process read function + if (slave->proc_read != NULL) { + slave->proc_read(slave, period); + } + } +} + +void lcec_write_master(void *arg, long period) { + lcec_master_t *master = (lcec_master_t *) arg; + lcec_slave_t *slave; + + // process slaves + for (slave = master->first_slave; slave != NULL; slave = slave->next) { + if (slave->proc_write != NULL) { + slave->proc_write(slave, period); + } + } + + // send process data + rtapi_mutex_get(&master->mutex); + + // update application time + master->app_time += master->app_time_period; + ecrt_master_application_time(master->master, master->app_time); + + // sync ref clock to master + if (master->sync_ref_cycles > 0) { + if (master->sync_ref_cnt == 0) { + master->sync_ref_cnt = master->sync_ref_cycles; + ecrt_master_sync_reference_clock(master->master); + } + master->sync_ref_cnt--; + } + + // sync slaves to ref clock + ecrt_master_sync_slave_clocks(master->master); + + // send domain data + ecrt_domain_queue(master->domain); + ecrt_master_send(master->master); + rtapi_mutex_give(&master->mutex); +} + +ec_sdo_request_t *lcec_read_sdo(struct lcec_slave *slave, uint16_t index, uint8_t subindex, size_t size) { + lcec_master_t *master = slave->master; + ec_sdo_request_t *sdo; + ec_request_state_t sdo_state; + unsigned long jiffies_start; + + // create request + if (!(sdo = ecrt_slave_config_create_sdo_request(slave->config, index, subindex, size))) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "slave %s.%s: Failed to create SDO request (0x%04x:0x%02x)\n", master->name, slave->name, index, subindex); + return NULL; + } + + // set timeout + ecrt_sdo_request_timeout(sdo, LCEC_SDO_REQ_TIMEOUT); + + // send request + ecrt_sdo_request_read(sdo); + + // wait for completition (master's time out does not work here. why???) + jiffies_start = jiffies; + while ((sdo_state = ecrt_sdo_request_state(sdo)) == EC_REQUEST_BUSY && (jiffies - jiffies_start) < (HZ * LCEC_SDO_REQ_TIMEOUT / 1000)) { + schedule(); + } + + // check state + if (sdo_state != EC_REQUEST_SUCCESS) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "slave %s.%s: Failed to execute SDO request (0x%04x:0x%02x)\n", master->name, slave->name, index, subindex); + return NULL; + } + + return sdo; +} + diff --git a/src/lcec_stmds5k.c b/src/lcec_stmds5k.c new file mode 100644 index 00000000..73543740 --- /dev/null +++ b/src/lcec_stmds5k.c @@ -0,0 +1,514 @@ +// +// Copyright (C) 2011 Sascha Ittner +// +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2 of the License, or +// (at your option) any later version. +// +// This program 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 this program; if not, write to the Free Software +// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA +// + +#include "lcec.h" +#include "lcec_stmds5k.h" + +#define STMDS5K_PCT_REG_FACTOR (0.5 * (double)0x7fff) +#define STMDS5K_PCT_REG_DIV (2.0 / (double)0x7fff) +#define STMDS5K_TORQUE_DIV (8.0 / (double)0x7fff) +#define STMDS5K_TORQUE_REF_DIV (0.01) +#define STMDS5K_RPM_FACTOR (60.0) +#define STMDS5K_RPM_DIV (1.0 / 60.0) + +typedef struct { + int do_init; + + long long pos_cnt; + long long index_cnt; + int32_t last_pos_cnt; + + hal_float_t *vel_cmd; + hal_float_t *vel_fb; + hal_float_t *vel_fb_rpm; + hal_float_t *vel_fb_rpm_abs; + hal_float_t *vel_rpm; + hal_u32_t *pos_raw_hi; + hal_u32_t *pos_raw_lo; + hal_float_t *pos_fb; + hal_float_t *torque_fb; + hal_float_t *torque_fb_abs; + hal_float_t *torque_fb_pct; + hal_float_t *torque_lim; + hal_bit_t *stopped; + hal_bit_t *at_speed; + hal_bit_t *overload; + hal_bit_t *ready; + hal_bit_t *error; + hal_bit_t *toggle; + hal_bit_t *loc_ena; + hal_bit_t *enable; + hal_bit_t *err_reset; + hal_bit_t *fast_ramp; + hal_bit_t *brake; + hal_bit_t *index_ena; + hal_bit_t *pos_reset; + hal_s32_t *enc_raw; + hal_bit_t *on_home_neg; + hal_bit_t *on_home_pos; + + hal_float_t speed_max_rpm; + hal_float_t speed_max_rpm_sp; + hal_float_t torque_reference; + hal_float_t pos_scale; + hal_s32_t home_raw; + double speed_max_rpm_sp_rcpt; + + double pos_scale_old; + double pos_scale_rcpt; + double pos_scale_cnt; + + int last_index_ena; + int32_t index_ref; + + unsigned int dev_state_pdo_os; + unsigned int speed_mot_pdo_os; + unsigned int torque_mot_pdo_os; + unsigned int speed_state_pdo_os; + unsigned int pos_mot_pdo_os; + unsigned int dev_ctrl_pdo_os; + unsigned int speed_sp_rel_pdo_os; + unsigned int torque_max_pdo_os; + + ec_sdo_request_t *sdo_torque_reference; + ec_sdo_request_t *sdo_speed_max_rpm; + ec_sdo_request_t *sdo_speed_max_rpm_sp; + +} lcec_stmds5k_data_t; + +void lcec_stmds5k_check_scales(lcec_stmds5k_data_t *hal_data); + +void lcec_stmds5k_read(struct lcec_slave *slave, long period); +void lcec_stmds5k_write(struct lcec_slave *slave, long period); + +int lcec_stmds5k_init(int comp_id, struct lcec_slave *slave, ec_pdo_entry_reg_t *pdo_entry_regs) { + lcec_master_t *master = slave->master; + lcec_stmds5k_data_t *hal_data; + int err; + + // initialize callbacks + slave->proc_read = lcec_stmds5k_read; + slave->proc_write = lcec_stmds5k_write; + + // alloc hal memory + if ((hal_data = hal_malloc(sizeof(lcec_stmds5k_data_t))) == NULL) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "hal_malloc() for slave %s.%s failed\n", master->name, slave->name); + return -EIO; + } + memset(hal_data, 0, sizeof(lcec_stmds5k_data_t)); + slave->hal_data = hal_data; + + // read sdos + // B18 : torque reference + if ((hal_data->sdo_torque_reference = lcec_read_sdo(slave, 0x2212, 0x00, 4)) == NULL) { + return -EIO; + } + hal_data->torque_reference = (double)EC_READ_S32(ecrt_sdo_request_data(hal_data->sdo_torque_reference)) * STMDS5K_TORQUE_REF_DIV; + // C01 : max rpm + if ((hal_data->sdo_speed_max_rpm = lcec_read_sdo(slave, 0x2401, 0x00, 4)) == NULL) { + return -EIO; + } + hal_data->speed_max_rpm = (double)EC_READ_S32(ecrt_sdo_request_data(hal_data->sdo_speed_max_rpm)); + // D02 : setpoint max rpm + if ((hal_data->sdo_speed_max_rpm_sp = lcec_read_sdo(slave, 0x2602, 0x00, 4)) == NULL) { + return -EIO; + } + hal_data->speed_max_rpm_sp = (double)EC_READ_S32(ecrt_sdo_request_data(hal_data->sdo_speed_max_rpm_sp)); + if (hal_data->speed_max_rpm_sp > 1e-20 || hal_data->speed_max_rpm_sp < -1e-20) { + hal_data->speed_max_rpm_sp_rcpt = 1.0 / hal_data->speed_max_rpm_sp; + } else { + hal_data->speed_max_rpm_sp_rcpt = 0.0; + } + + // initialize POD entries + // E200 : device state byte + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x28c8, 0x00, &hal_data->dev_state_pdo_os, NULL); + // E100 : speed motor (x 0.1% relative to C01) + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x2864, 0x00, &hal_data->speed_mot_pdo_os, NULL); + // E02 : torque motor filterd (x 0,1 Nm) + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x2802, 0x00, &hal_data->torque_mot_pdo_os, NULL); + // D200 : speed state word + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x26c8, 0x00, &hal_data->speed_state_pdo_os, NULL); + // E09 : rotor position + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x2809, 0x00, &hal_data->pos_mot_pdo_os, NULL); + // A180 : Device Control Byte + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x20b4, 0x00, &hal_data->dev_ctrl_pdo_os, NULL); + // D230 : speed setpoint (x 0.1 % relative to D02, -200.0% .. 200.0%) + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x26e6, 0x00, &hal_data->speed_sp_rel_pdo_os, NULL); + // C230 : maximum torque (x 1%, 0% .. 200%) + LCEC_PDO_INIT(pdo_entry_regs, slave->index, slave->vid, slave->pid, 0x24e6, 0x00, &hal_data->torque_max_pdo_os, NULL); + + // export pins + if ((err = hal_pin_float_newf(HAL_IN, &(hal_data->vel_cmd), comp_id, "%s.%s.%s.srv-vel-cmd", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-vel-cmd failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_float_newf(HAL_OUT, &(hal_data->vel_fb), comp_id, "%s.%s.%s.srv-vel-fb", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-vel-fb failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_float_newf(HAL_OUT, &(hal_data->vel_fb_rpm), comp_id, "%s.%s.%s.srv-vel-fb-rpm", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-vel-fb-rpm failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_float_newf(HAL_OUT, &(hal_data->vel_fb_rpm_abs), comp_id, "%s.%s.%s.srv-vel-fb-rpm-abs", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-vel-fb-rpm-abs failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_float_newf(HAL_OUT, &(hal_data->vel_rpm), comp_id, "%s.%s.%s.srv-vel-rpm", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-vel-rpm failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_s32_newf(HAL_OUT, &(hal_data->enc_raw), comp_id, "%s.%s.%s.srv-enc-raw", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-enc-raw failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_u32_newf(HAL_OUT, &(hal_data->pos_raw_hi), comp_id, "%s.%s.%s.srv-pos-raw-hi", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-pos-raw-hi failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_u32_newf(HAL_OUT, &(hal_data->pos_raw_lo), comp_id, "%s.%s.%s.srv-pos-raw-lo", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-pos-raw-lo failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_float_newf(HAL_OUT, &(hal_data->pos_fb), comp_id, "%s.%s.%s.srv-pos-fb", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-pos-fb failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_float_newf(HAL_OUT, &(hal_data->torque_fb), comp_id, "%s.%s.%s.srv-torque-fb", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-torque-fb failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_float_newf(HAL_OUT, &(hal_data->torque_fb_abs), comp_id, "%s.%s.%s.srv-torque-fb-abs", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-torque-fb-abs failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_float_newf(HAL_OUT, &(hal_data->torque_fb_pct), comp_id, "%s.%s.%s.srv-torque-fb-pct", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-torque-fb-pct failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_float_newf(HAL_IN, &(hal_data->torque_lim), comp_id, "%s.%s.%s.srv-torque-lim", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-torque-lim failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(hal_data->stopped), comp_id, "%s.%s.%s.srv-stopped", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-stopped failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(hal_data->at_speed), comp_id, "%s.%s.%s.srv-at-speed", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-at-speed failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(hal_data->overload), comp_id, "%s.%s.%s.srv-overload", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-overload failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(hal_data->ready), comp_id, "%s.%s.%s.srv-ready", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-ready failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(hal_data->error), comp_id, "%s.%s.%s.srv-error", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-error failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(hal_data->toggle), comp_id, "%s.%s.%s.srv-toggle", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-toggle failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(hal_data->loc_ena), comp_id, "%s.%s.%s.srv-loc-ena", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-loc-ena failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_IN, &(hal_data->enable), comp_id, "%s.%s.%s.srv-enable", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-enable failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_IN, &(hal_data->err_reset), comp_id, "%s.%s.%s.srv-err-reset", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-err-reset failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_IN, &(hal_data->fast_ramp), comp_id, "%s.%s.%s.srv-fast-ramp", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-fast-ramp failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_IN, &(hal_data->brake), comp_id, "%s.%s.%s.srv-brake", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-brake failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_IO, &(hal_data->index_ena), comp_id, "%s.%s.%s.srv-index-ena", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-index-ena failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_IN, &(hal_data->pos_reset), comp_id, "%s.%s.%s.srv-pos-reset", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-pos-reset failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(hal_data->on_home_neg), comp_id, "%s.%s.%s.srv-on-home-neg", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-on-home-neg failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_pin_bit_newf(HAL_OUT, &(hal_data->on_home_pos), comp_id, "%s.%s.%s.srv-on-home-pos", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-on-home-pos failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + + // export parameters + if ((err = hal_param_float_newf(HAL_RW, &(hal_data->pos_scale), comp_id, "%s.%s.%s.srv-pos-scale", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-pos-scale failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_param_float_newf(HAL_RO, &(hal_data->torque_reference), comp_id, "%s.%s.%s.srv-torque-ref", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-torque-ref failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_param_float_newf(HAL_RO, &(hal_data->speed_max_rpm), comp_id, "%s.%s.%s.srv-max-rpm", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-max-rpm failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_param_float_newf(HAL_RO, &(hal_data->speed_max_rpm_sp), comp_id, "%s.%s.%s.srv-max-rpm-sp", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-max-rpm-sp failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + if ((err = hal_param_s32_newf(HAL_RW, &(hal_data->home_raw), comp_id, "%s.%s.%s.srv-home-raw", LCEC_MODULE_NAME, master->name, slave->name)) != 0) { + rtapi_print_msg(RTAPI_MSG_ERR, LCEC_MSG_PFX "exporting pin %s.%s.%s.srv-home-raw failed\n", LCEC_MODULE_NAME, master->name, slave->name); + return err; + } + + // set default pin values + *(hal_data->vel_cmd) = 0.0; + *(hal_data->vel_fb) = 0.0; + *(hal_data->vel_fb_rpm) = 0.0; + *(hal_data->vel_fb_rpm_abs) = 0.0; + *(hal_data->vel_rpm) = 0.0; + *(hal_data->pos_raw_hi) = 0; + *(hal_data->pos_raw_lo) = 0; + *(hal_data->pos_fb) = 0.0; + *(hal_data->torque_fb) = 0.0; + *(hal_data->torque_fb_abs) = 0.0; + *(hal_data->torque_fb_pct) = 0.0; + *(hal_data->torque_lim) = 1.0; + *(hal_data->stopped) = 0; + *(hal_data->at_speed) = 0; + *(hal_data->overload) = 0; + *(hal_data->ready) = 0; + *(hal_data->error) = 0; + *(hal_data->toggle) = 0; + *(hal_data->loc_ena) = 0; + *(hal_data->enable) = 0; + *(hal_data->err_reset) = 0; + *(hal_data->fast_ramp) = 0; + *(hal_data->brake) = 0; + *(hal_data->index_ena) = 0; + *(hal_data->pos_reset) = 0; + *(hal_data->enc_raw) = 0; + *(hal_data->on_home_neg) = 0; + *(hal_data->on_home_pos) = 0; + + // initialize variables + hal_data->pos_scale = 1.0; + hal_data->do_init = 1; + hal_data->pos_cnt = 0; + hal_data->index_cnt = 0; + hal_data->last_pos_cnt = 0; + hal_data->pos_scale_old = hal_data->pos_scale + 1.0; + hal_data->pos_scale_rcpt = 1.0; + hal_data->pos_scale_cnt = 1.0; + hal_data->last_index_ena = 0; + hal_data->index_ref = 0; + hal_data->home_raw = 0; + + return 0; +} + +void lcec_stmds5k_check_scales(lcec_stmds5k_data_t *hal_data) { + // check for change in scale value + if (hal_data->pos_scale != hal_data->pos_scale_old) { + // scale value has changed, test and update it + if ((hal_data->pos_scale < 1e-20) && (hal_data->pos_scale > -1e-20)) { + // value too small, divide by zero is a bad thing + hal_data->pos_scale = 1.0; + } + // save new scale to detect future changes + hal_data->pos_scale_old = hal_data->pos_scale; + // we actually want the reciprocal + hal_data->pos_scale_rcpt = 1.0 / hal_data->pos_scale; + // scale for counter + hal_data->pos_scale_cnt = hal_data->pos_scale_rcpt / (double)(0x100000000LL); + } +} + +void lcec_stmds5k_read(struct lcec_slave *slave, long period) { + lcec_master_t *master = slave->master; + lcec_stmds5k_data_t *hal_data = (lcec_stmds5k_data_t *) slave->hal_data; + uint8_t *pd = master->process_data; + int32_t index_tmp; + int32_t pos_cnt, pos_cnt_diff; + long long net_count; + uint8_t dev_state; + uint16_t speed_state; + int16_t speed_raw, torque_raw; + double rpm, torque; + + // wait for slave to be operational + if (!slave->state.operational) { + return; + } + + // check for change in scale value + lcec_stmds5k_check_scales(hal_data); + + // read device state + dev_state = EC_READ_U8(&pd[hal_data->dev_state_pdo_os]); + *(hal_data->ready) = (dev_state >> 0) & 0x01; + *(hal_data->error) = (dev_state >> 1) & 0x01; + *(hal_data->loc_ena) = (dev_state >> 6) & 0x01; + *(hal_data->toggle) = (dev_state >> 7) & 0x01; + + // read speed state + speed_state = EC_READ_U16(&pd[hal_data->speed_state_pdo_os]); + *(hal_data->stopped) = (speed_state >> 0) & 0x01; + *(hal_data->at_speed) = (speed_state >> 1) & 0x01; + *(hal_data->overload) = (speed_state >> 2) & 0x01; + + // read current speed + speed_raw = EC_READ_S16(&pd[hal_data->speed_mot_pdo_os]); + rpm = hal_data->speed_max_rpm * (double)speed_raw * STMDS5K_PCT_REG_DIV; + *(hal_data->vel_fb_rpm) = rpm; + *(hal_data->vel_fb_rpm_abs) = fabs(rpm); + *(hal_data->vel_fb) = rpm * STMDS5K_RPM_DIV * hal_data->pos_scale_rcpt; + + // read torque + // E02 : torque motor filterd (x 0,1 Nm) + torque_raw = EC_READ_S16(&pd[hal_data->torque_mot_pdo_os]); + torque = (double)torque_raw * STMDS5K_TORQUE_DIV; + *(hal_data->torque_fb_pct) = torque * 100.0; + torque = torque * hal_data->torque_reference; + *(hal_data->torque_fb) = torque; + torque = fabs(torque); + *(hal_data->torque_fb_abs) = torque; + + // update raw position counter + pos_cnt = EC_READ_S32(&pd[hal_data->pos_mot_pdo_os]); + *(hal_data->enc_raw) = pos_cnt; + *(hal_data->on_home_neg) = ((pos_cnt - hal_data->home_raw) <= 0); + *(hal_data->on_home_pos) = ((pos_cnt - hal_data->home_raw) >= 0); + pos_cnt <<= 8; + pos_cnt_diff = pos_cnt - hal_data->last_pos_cnt; + hal_data->last_pos_cnt = pos_cnt; + hal_data->pos_cnt += pos_cnt_diff; + + // check for index edge + if (*(hal_data->index_ena)) { + index_tmp = (hal_data->pos_cnt >> 32) & 0xffffffff; + if (hal_data->do_init || !hal_data->last_index_ena) { + hal_data->index_ref = index_tmp; + } else if (index_tmp > hal_data->index_ref) { + hal_data->index_cnt = (long long)index_tmp << 32; + *(hal_data->index_ena) = 0; + } else if (index_tmp < hal_data->index_ref) { + hal_data->index_cnt = (long long)hal_data->index_ref << 32; + *(hal_data->index_ena) = 0; + } + } + hal_data->last_index_ena = *(hal_data->index_ena); + + // handle initialization + if (hal_data->do_init || *(hal_data->pos_reset)) { + hal_data->do_init = 0; + hal_data->index_cnt = hal_data->pos_cnt; + } + + // compute net counts + net_count = hal_data->pos_cnt - hal_data->index_cnt; + + // update raw counter pins + *(hal_data->pos_raw_hi) = (net_count >> 32) & 0xffffffff; + *(hal_data->pos_raw_lo) = net_count & 0xffffffff; + + // scale count to make floating point position + *(hal_data->pos_fb) = net_count * hal_data->pos_scale_cnt; +} + +void lcec_stmds5k_write(struct lcec_slave *slave, long period) { + lcec_master_t *master = slave->master; + lcec_stmds5k_data_t *hal_data = (lcec_stmds5k_data_t *) slave->hal_data; + uint8_t *pd = master->process_data; + uint8_t dev_ctrl; + double speed_raw, torque_raw; + + // check for change in scale value + lcec_stmds5k_check_scales(hal_data); + + // write dev ctrl + dev_ctrl = 0; + if (*(hal_data->enable)) { + dev_ctrl |= (1 << 0); + } + if (*(hal_data->err_reset)) { + dev_ctrl |= (1 << 1); + } + if (*(hal_data->fast_ramp)) { + dev_ctrl |= (1 << 2); + } + if (*(hal_data->brake)) { + dev_ctrl |= (1 << 6); + } + if (! *(hal_data->toggle)) { + dev_ctrl |= (1 << 7); + } + EC_WRITE_U8(&pd[hal_data->dev_ctrl_pdo_os], dev_ctrl); + + // set maximum torque + if (*(hal_data->torque_lim) > 2.0) { + *(hal_data->torque_lim) = 2.0; + } + if (*(hal_data->torque_lim) < -2.0) { + *(hal_data->torque_lim) = -2.0; + } + torque_raw = *(hal_data->torque_lim) * STMDS5K_PCT_REG_FACTOR; + if (torque_raw > (double)0x7fff) { + torque_raw = (double)0x7fff; + } + if (torque_raw < (double)-0x7fff) { + torque_raw = (double)-0x7fff; + } + EC_WRITE_S16(&pd[hal_data->torque_max_pdo_os], (int16_t)torque_raw); + + // calculate rpm command + *(hal_data->vel_rpm) = *(hal_data->vel_cmd) * hal_data->pos_scale * STMDS5K_RPM_FACTOR; + + // set RPM + if (*(hal_data->vel_rpm) > hal_data->speed_max_rpm) { + *(hal_data->vel_rpm) = hal_data->speed_max_rpm; + } + if (*(hal_data->vel_rpm) < -hal_data->speed_max_rpm) { + *(hal_data->vel_rpm) = -hal_data->speed_max_rpm; + } + speed_raw = *(hal_data->vel_rpm) * hal_data->speed_max_rpm_sp_rcpt * STMDS5K_PCT_REG_FACTOR; + if (speed_raw > (double)0x7fff) { + speed_raw = (double)0x7fff; + } + if (speed_raw < (double)-0x7fff) { + speed_raw = (double)-0x7fff; + } + if (! *(hal_data->enable)) { + speed_raw = 0.0; + } + EC_WRITE_S16(&pd[hal_data->speed_sp_rel_pdo_os], (int16_t)speed_raw); +} + diff --git a/src/lcec_stmds5k.h b/src/lcec_stmds5k.h new file mode 100644 index 00000000..0353f7c9 --- /dev/null +++ b/src/lcec_stmds5k.h @@ -0,0 +1,31 @@ +// +// Copyright (C) 2011 Sascha Ittner +// +// This program is free software; you can redistribute it and/or modify +// it under the terms of the GNU General Public License as published by +// the Free Software Foundation; either version 2 of the License, or +// (at your option) any later version. +// +// This program 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 this program; if not, write to the Free Software +// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA +// +#ifndef _LCEC_STMDS5K_H_ +#define _LCEC_STMDS5K_H_ + +#include "lcec.h" + +#define LCEC_STMDS5K_VID LCEC_STOEBER_VID +#define LCEC_STMDS5K_PID 0x00001388 + +#define LCEC_STMDS5K_PDOS 8 + +int lcec_stmds5k_init(int comp_id, struct lcec_slave *slave, ec_pdo_entry_reg_t *pdo_entry_regs); + +#endif + diff --git a/src/lcnc_rtapi.h b/src/lcnc_rtapi.h new file mode 100644 index 00000000..59d2f8b3 --- /dev/null +++ b/src/lcnc_rtapi.h @@ -0,0 +1,31 @@ +/******************************************************************** + * fake unified build config_module.h + *******************************************************************/ + +#ifndef _LCEC_RTAPI_H_ +#define _LCEC_RTAPI_H_ + +//#include "config_module.h" + +#define RTAPI_INC_CTYPE_H "linux/ctype.h" +#define RTAPI_INC_DEVICE_H "linux/device.h" +#define RTAPI_INC_FIRMWARE_H "linux/firmware.h" +#define RTAPI_INC_GFP_H "linux/gfp.h" +#define RTAPI_INC_LIST_H "linux/list.h" +#define RTAPI_INC_SLAB_H "linux/slab.h" +#define RTAPI_INC_STRING_H "linux/string.h" +#define RTAPI_INC_SCHED_H "linux/sched.h" +#define RTAPI_INC_JIFFIES_H "linux/jiffies.h" +#define RTAPI_INC_TIME_H "linux/time.h" + +#define RTAPI_GETTIMEOFDAY(x) do_gettimeofday(x) +//#define RTAPI_GETTIMEOFDAY(x) gettimeofday(x, NULL) + +#include RTAPI_INC_SCHED_H +#include RTAPI_INC_JIFFIES_H +#include RTAPI_INC_CTYPE_H +#include RTAPI_INC_SLAB_H +#include RTAPI_INC_STRING_H +#include RTAPI_INC_TIME_H + +#endif /* CONFIG_MODULE_H */