# HG changeset patch
# User Martin Troxler <martin.troxler@komaxgroup.com>
# Date 1259580416 -3600
# Node ID 1f640e321ee40b12e1b17d8f239e014011f3fea4
# Parent  cd9a68fd5b89a8f854ae57c479b6272eb1899843
ecrt_master_set_max_cycle_size: limit SDO traffic

diff -r cd9a68fd5b89 -r 1f640e321ee4 include/ecrt.h
--- a/include/ecrt.h	Mon Nov 30 12:26:24 2009 +0100
+++ b/include/ecrt.h	Mon Nov 30 12:26:56 2009 +0100
@@ -698,6 +698,15 @@
         ec_master_t *master /**< EtherCAT master. */
         );
 
+
+/** Set max. number of databytes in a cycle
+ *
+ */
+int ecrt_master_set_max_cycle_size(
+        ec_master_t *master, /**< EtherCAT master. */
+        size_t max_cycle_data_size /**< Max. number of databytes in a cycle */
+        );
+
 /** Sends all datagrams in the queue.
  *
  * This method takes all datagrams, that have been queued for transmission,
diff -r cd9a68fd5b89 -r 1f640e321ee4 lib/master.c
--- a/lib/master.c	Mon Nov 30 12:26:24 2009 +0100
+++ b/lib/master.c	Mon Nov 30 12:26:56 2009 +0100
@@ -344,6 +344,21 @@
     }
 }
 
+
+/*****************************************************************************/
+
+int ecrt_master_set_max_cycle_size(ec_master_t *master,size_t max_cycle_data_size)
+{
+    if (ioctl(master->fd, EC_IOCTL_SET_MAX_CYCLE_SIZE,
+                &max_cycle_data_size) == -1) {
+        fprintf(stderr, "Failed to activate master: %s\n",
+                strerror(errno));
+        return -1; // FIXME
+    }
+    return 0;
+}
+
+
 /*****************************************************************************/
 
 void ecrt_master_send(ec_master_t *master)
diff -r cd9a68fd5b89 -r 1f640e321ee4 master/cdev.c
--- a/master/cdev.c	Mon Nov 30 12:26:24 2009 +0100
+++ b/master/cdev.c	Mon Nov 30 12:26:56 2009 +0100
@@ -1673,6 +1673,32 @@
     return 0;
 }
 
+
+/*****************************************************************************/
+
+/** Set max. number of databytes in a cycle
+ */
+int ec_cdev_ioctl_set_max_cycle_size(
+        ec_master_t *master, /**< EtherCAT master. */
+        unsigned long arg, /**< ioctl() argument. */
+        ec_cdev_priv_t *priv /**< Private data structure of file handle. */
+        )
+{
+    size_t max_cycle_size;
+
+    if (copy_from_user(&max_cycle_size, (void __user *) arg, sizeof(max_cycle_size))) {
+        return -EFAULT;
+    }
+
+    if (down_interruptible(&master->master_sem))
+        return -EINTR;
+    master->max_queue_size = max_cycle_size;
+    up(&master->master_sem);
+
+    return 0;
+}
+
+
 /*****************************************************************************/
 
 /** Send frames.
@@ -3466,6 +3492,10 @@
             return ec_cdev_ioctl_voe_exec(master, arg, priv);
         case EC_IOCTL_VOE_DATA:
             return ec_cdev_ioctl_voe_data(master, arg, priv);
+        case EC_IOCTL_SET_MAX_CYCLE_SIZE:
+            if (!(filp->f_mode & FMODE_WRITE))
+                return -EPERM;
+            return ec_cdev_ioctl_set_max_cycle_size(master,arg,priv);
         default:
             return -ENOTTY;
     }
diff -r cd9a68fd5b89 -r 1f640e321ee4 master/globals.h
--- a/master/globals.h	Mon Nov 30 12:26:24 2009 +0100
+++ b/master/globals.h	Mon Nov 30 12:26:56 2009 +0100
@@ -48,6 +48,12 @@
 /** Datagram timeout in microseconds. */
 #define EC_IO_TIMEOUT 500
 
+/** SDO injection timeout in microseconds. */
+#define EC_SDO_INJECTION_TIMEOUT 10000
+
+/** time to send a byte in nanoseconds. */
+#define EC_BYTE_TRANSMITION_TIME 80LL
+
 /** Number of state machine retries on datagram timeout. */
 #define EC_FSM_RETRIES 3
 
diff -r cd9a68fd5b89 -r 1f640e321ee4 master/ioctl.h
--- a/master/ioctl.h	Mon Nov 30 12:26:24 2009 +0100
+++ b/master/ioctl.h	Mon Nov 30 12:26:56 2009 +0100
@@ -123,6 +123,7 @@
 #define EC_IOCTL_VOE_WRITE            EC_IOWR(0x3f, ec_ioctl_voe_t)
 #define EC_IOCTL_VOE_EXEC             EC_IOWR(0x40, ec_ioctl_voe_t)
 #define EC_IOCTL_VOE_DATA             EC_IOWR(0x41, ec_ioctl_voe_t)
+#define EC_IOCTL_SET_MAX_CYCLE_SIZE    EC_IOW(0x42, size_t)
 
 /*****************************************************************************/
 
diff -r cd9a68fd5b89 -r 1f640e321ee4 master/master.c
--- a/master/master.c	Mon Nov 30 12:26:24 2009 +0100
+++ b/master/master.c	Mon Nov 30 12:26:56 2009 +0100
@@ -59,13 +59,14 @@
 /** Frame timeout in cycles.
  */
 static cycles_t timeout_cycles;
-
+static cycles_t sdo_injection_timeout_cycles;
 #else
 
 /** Frame timeout in jiffies.
  */
 static unsigned long timeout_jiffies;
-    
+static unsigned long sdo_injection_timeout_jiffies;
+
 #endif
 
 /*****************************************************************************/
@@ -87,9 +88,11 @@
 {
 #ifdef EC_HAVE_CYCLES
     timeout_cycles = (cycles_t) EC_IO_TIMEOUT /* us */ * (cpu_khz / 1000);
+    sdo_injection_timeout_cycles = (cycles_t) EC_SDO_INJECTION_TIMEOUT /* us */ * (cpu_khz / 1000);
 #else
     // one jiffy may always elapse between time measurement
     timeout_jiffies = max(EC_IO_TIMEOUT * HZ / 1000000, 1);
+    sdo_injection_timeout_jiffies = max(EC_SDO_INJECTION_TIMEOUT * HZ / 1000000, 1);
 #endif
 }
 
@@ -152,6 +155,7 @@
     sema_init(&master->ext_queue_sem, 1);
 
     INIT_LIST_HEAD(&master->sdo_datagram_queue);
+    master->max_queue_size = EC_MAX_DATA_SIZE;
 
     INIT_LIST_HEAD(&master->domains);
 
@@ -695,12 +699,56 @@
         ec_master_t *master /**< EtherCAT master */
         )
 {
-    ec_datagram_t *sdo_datagram, *n;
-    list_for_each_entry_safe(sdo_datagram, n, &master->sdo_datagram_queue, queue) {
-        list_del_init(&sdo_datagram->queue);
-        if (master->debug_level)
-            EC_DBG("queuing SDO datagram %08x\n",(unsigned int)sdo_datagram);
-        ec_master_queue_datagram(master, sdo_datagram);
+    ec_datagram_t *datagram, *n;
+    size_t queue_size = 0;
+    list_for_each_entry(datagram, &master->datagram_queue, queue) {
+        queue_size += datagram->data_size;
+    }
+    list_for_each_entry_safe(datagram, n, &master->sdo_datagram_queue, queue) {
+        queue_size += datagram->data_size;
+        if (queue_size <= master->max_queue_size) {
+            list_del_init(&datagram->queue);
+            if (master->debug_level) {
+                EC_DBG("Injecting SDO datagram %08x size=%u, queue_size=%u\n",(unsigned int)datagram,datagram->data_size,queue_size);
+            }
+#ifdef EC_HAVE_CYCLES
+            datagram->cycles_sent = 0;
+#endif
+            datagram->jiffies_sent = 0;
+            ec_master_queue_datagram(master, datagram);
+        }
+        else {
+            if (datagram->data_size > master->max_queue_size) {
+                list_del_init(&datagram->queue);
+                datagram->state = EC_DATAGRAM_ERROR;
+                EC_ERR("SDO datagram %08x is too large, size=%u, max_queue_size=%u\n",(unsigned int)datagram,datagram->data_size,master->max_queue_size);
+            }
+            else {
+#ifdef EC_HAVE_CYCLES
+                cycles_t cycles_now = get_cycles();
+                if (cycles_now - datagram->cycles_sent
+                        > sdo_injection_timeout_cycles) {
+#else
+                if (jiffies - datagram->jiffies_sent
+                        > sdo_injection_timeout_jiffies) {
+#endif
+                    unsigned int time_us;
+                    list_del_init(&datagram->queue);
+                    datagram->state = EC_DATAGRAM_ERROR;
+#ifdef EC_HAVE_CYCLES
+                    time_us = (unsigned int) ((cycles_now - datagram->cycles_sent) * 1000LL) / cpu_khz;
+#else
+                    time_us = (unsigned int) ((jiffies - datagram->jiffies_sent) * 1000000 / HZ);
+#endif
+                    EC_ERR("Timeout %u us: injecting SDO datagram %08x size=%u, max_queue_size=%u\n",time_us,(unsigned int)datagram,datagram->data_size,master->max_queue_size);
+                }
+                else  {
+                    if (master->debug_level) {
+                        EC_DBG("Deferred injecting of SDO datagram %08x size=%u, queue_size=%u\n",(unsigned int)datagram,datagram->data_size,queue_size);
+                    }
+                }
+            }
+        }
     }
 }
 
@@ -713,9 +761,17 @@
         ec_datagram_t *datagram /**< datagram */
         )
 {
+    if (master->debug_level) {
+        EC_DBG("Requesting SDO datagram %08x size=%u\n",(unsigned int)datagram,datagram->data_size);
+    }
+    datagram->state = EC_DATAGRAM_QUEUED;
+#ifdef EC_HAVE_CYCLES
+    datagram->cycles_sent = get_cycles();
+#endif
+    datagram->jiffies_sent = jiffies;
+
     down(&master->io_sem);
     list_add_tail(&datagram->queue, &master->sdo_datagram_queue);
-    datagram->state = EC_DATAGRAM_QUEUED;
     up(&master->io_sem);
 }
 
diff -r cd9a68fd5b89 -r 1f640e321ee4 master/master.h
--- a/master/master.h	Mon Nov 30 12:26:24 2009 +0100
+++ b/master/master.h	Mon Nov 30 12:26:56 2009 +0100
@@ -155,7 +155,7 @@
                                       ext_datagram_queue. */
 
     struct list_head sdo_datagram_queue; /**< SDO Datagram queue. */
-
+    size_t max_queue_size; /** max. size of datagram queue */
     struct list_head domains; /**< List of domains. */
 
     unsigned int debug_level; /**< Master debug level. */