patch-2.2.14 linux/drivers/s390/net/ctc.c

Next file: linux/drivers/s390/net/iucv.c
Previous file: linux/drivers/s390/net/Makefile
Back to the patch index
Back to the overall index

diff -u --recursive --new-file v2.2.13/linux/drivers/s390/net/ctc.c linux/drivers/s390/net/ctc.c
@@ -0,0 +1,1408 @@
+/*
+ *  drivers/s390/net/ctc.c 
+ *    CTC / ESCON network driver
+ *
+ *  S390 version
+ *    Copyright (C) 1999 IBM Deutschland Entwicklung GmbH, IBM Corporation
+ *    Author(s): Dieter Wellerdiek (wel@de.ibm.com)
+ */
+
+#include <linux/kernel.h>
+#include <linux/malloc.h>
+#include <linux/errno.h>
+#include <linux/types.h>
+#include <linux/interrupt.h>
+#include <linux/timer.h>
+#include <linux/sched.h>
+
+#include <linux/signal.h>
+#include <linux/string.h>
+
+#include <linux/netdevice.h>
+#include <linux/etherdevice.h>
+#include <linux/ip.h>
+#include <linux/if_arp.h>
+#include <linux/tcp.h>
+#include <linux/skbuff.h>
+
+#include <asm/io.h>
+#include <asm/bitops.h>
+ 
+#include "../../../arch/s390/kernel/irq.h"
+
+
+//#define DEBUG 
+
+/* Redefine message level, so that all messages occure on 3215 console in DEBUG mode */
+#ifdef DEBUG                
+        #undef  KERN_INFO
+        #undef  KERN_WARNING
+        #undef  KERN_DEBUG
+        #define KERN_INFO    KERN_EMERG
+        #define KERN_WARNING KERN_EMERG
+        #define KERN_DEBUG   KERN_EMERG
+#endif  
+//#undef DEBUG
+
+#define CCW_CMD_WRITE           0x01
+#define CCW_CMD_READ            0x02
+#define CCW_CMD_SET_EXTENDED    0xc3
+#define CCW_CMD_PREPARE         0xe3
+
+#define MAX_DEVICES             16 
+#define MAX_ADAPTERS            MAX_DEVICES / 2
+#define CTC_DEFAULT_MTU_SIZE    1500
+#define READ                    0
+#define WRITE                   1
+#define CTC                     0
+#define ESCON                   1
+#define CHANNEL_MEDIA           2
+#define CTC_BLOCKS              8          /* 8 blocks * 2 times * 64k = 1M */
+
+#define TB_TX                   0          /* sk buffer handling in process */
+#define TB_STOP                 1          /* network device stop in process */
+#define TB_RETRY                2          /* retry in process */
+#define TB_NOBUFFER             3          /* no buffer on free queue */ 
+
+/* state machine codes used in ctc_irq_handler */
+#define CTC_STOP                0
+#define CTC_START_HALT_IO       1
+#define CTC_START_SET_X_MODE    2
+#define CTC_START_SELECT        4 
+#define CTC_START_READ_TEST     32
+#define CTC_START_READ          33
+#define CTC_START_WRITE_TEST    64
+#define CTC_START_WRITE         65
+
+
+typedef enum { 
+        channel_type_none,           /* Device is not a channel */
+        channel_type_undefined,      /* Device is a channel but we dont know anything about it */
+        channel_type_ctca,           /* Device is a CTC/A and we can deal with it */
+        channel_type_escon,          /* Device is a ESCON channel and we can deal with it */
+        channel_type_unsupported     /* Device is a unsupported model */
+} channel_type_t; 
+  
+
+
+/* 
+ *   Structures needed in the initial phase 
+ *
+ */  
+
+static int channel_tab_initialized = 0;     /* channel[] structure initialized */
+
+struct devicelist {  
+        unsigned int  devno;
+        __u8          flag;
+#define CHANNEL_IN_USE   0x08               /* - Show that channel is in use */
+}; 
+
+static struct {
+        struct devicelist  list[MAX_DEVICES]; 
+        int                count;
+        int                left;
+} channel[CHANNEL_MEDIA];
+
+
+
+static int ctc_no_auto = 0;
+
+struct adapterlist{ 
+        unsigned int       devno[2];
+        __u16              protocol;
+};
+
+static struct adapterlist ctc_adapter[CHANNEL_MEDIA][MAX_ADAPTERS];
+
+
+/* 
+ *   Structure used after the initial phase 
+ *
+ */
+    
+struct buffer {
+        struct buffer       *next;
+        int                 packets;
+        struct block        *block;
+};
+
+
+struct channel {
+        unsigned int        devno;
+        int                 irq;
+        int                 IO_active;
+        ccw1_t              ccw[3];
+        __u32               state; 
+        int                 buffer_count;
+        struct buffer       *free_anchor;
+        struct buffer       *proc_anchor;
+        devstat_t           *devstat;
+        struct device       *dev;      /* backward pointer to the network device */ 
+        struct wait_queue   *wait;
+        struct tq_struct    tq;
+        struct timer_list   timer;
+        unsigned long       flag_a;    /* atomic flags */
+#define CTC_BH_ACTIVE       0   
+        __u8                last_dstat;
+        __u8                flag;
+#define CTC_WRITE            0x01      /* - Set if this is a write channel */
+#define CTC_TIMER            0x80      /* - Set if timer made the wake_up  */ 
+};
+
+
+struct ctc_priv {                                                                    
+        struct net_device_stats  stats;
+        struct channel           channel[2]; 
+        __u16                    protocol;
+};  
+
+/*
+ *   This structure works as shuttle between two systems 
+ *    - A block can contain one or more packets 
+ */
+
+#define PACKET_HEADER_LENGTH  6
+struct packet {
+        __u16         length;
+        __u16         type;
+        __u16         unused;
+        __u8          data;
+}; 
+
+#define BLOCK_HEADER_LENGTH   2
+struct block {
+        __u16         length;
+        struct packet data;
+};
+
+
+/* Interrupt handler */
+static void ctc_irq_handler(int irq, void *initparm, struct pt_regs *regs);
+static void ctc_irq_bh(struct channel *ctc); 
+static void ctc_read_retry (struct channel *ctc);
+static void ctc_write_retry (struct channel *ctc);
+
+
+/* Functions for the DEV methods */
+void ctc_setup(char *dev_name, int *ints);
+int ctc_probe(struct device *dev);
+ 
+
+static int ctc_open(struct device *dev); 
+static void ctc_timer (struct channel *ctc);
+static int ctc_release(struct device *dev);
+static int ctc_tx(struct sk_buff *skb, struct device *dev);
+static int ctc_change_mtu(struct device *dev, int new_mtu);
+struct net_device_stats* ctc_stats(struct device *dev); 
+
+
+/*
+ *   Channel Routines 
+ *
+ */ 
+
+static void channel_init(void);
+static void channel_scan(void);
+static int channel_get(int media, int devno);
+static int channel_get_next(int media); 
+static int channel_free(int media, int devno);
+static channel_type_t channel_check_for_type (senseid_t *id);
+static void channel_sort(struct devicelist list[], int n);
+
+
+/*
+ * initialize the channel[].list 
+ */   
+static void channel_init(void) 
+{
+        int     m;
+#ifdef DEBUG
+        int     c;
+#endif
+
+        if (!test_and_set_bit(0, (void *)& channel_tab_initialized)){
+                channel_scan(); 
+                for (m = 0; m < CHANNEL_MEDIA; m++) { 
+                        channel_sort (channel[m].list, MAX_DEVICES); 
+                        channel[m].left = channel[m].count;   
+                }
+                if (channel[CTC].count == 0 && channel[ESCON].count == 0) 
+                        printk(KERN_INFO "channel: no Channel devices recognized\n");
+                else
+                        printk(KERN_INFO "channel: %d Parallel channel found - %d ESCON channel found\n",
+                            channel[CTC].count, channel[ESCON].count);  
+#ifdef DEBUG 
+                for (m = 0; m < CHANNEL_MEDIA;  m++) { 
+                        for (c = 0; c < MAX_DEVICES; c++){
+                                printk(KERN_DEBUG "channel: Adapter=%x Entry=%x devno=%04x\n", 
+                                     m, c, channel[m].list[c].devno);
+                        }
+                }
+#endif
+         }
+}
+
+
+/*
+* scan for all channels and put the device numbers into the channel[].list 
+*/  
+static void channel_scan(void)
+{
+        int        m;
+        int        c;
+        int        irq;
+        dev_info_t temp;
+        
+        for (m = 0; m < CHANNEL_MEDIA;  m++) { 
+                for (c = 0; c < MAX_DEVICES; c++){
+                        channel[m].list[c].devno = -ENODEV;
+                }
+        }
+        
+        for (irq = 0; irq < NR_IRQS; irq++) {
+                /* CTC/A */
+                if (channel[CTC].count < MAX_DEVICES ) {
+                        if (get_dev_info(irq, &temp) == 0 && 
+                            channel_check_for_type(&temp.sid_data) == channel_type_ctca) {
+                                channel[CTC].list[channel[CTC].count].devno = temp.devno; 
+                                channel[CTC].count++; 
+                        }
+                }
+
+                /* ESCON */
+                if (channel[ESCON].count < MAX_DEVICES ) {
+                        if (get_dev_info(irq, &temp) == 0 && 
+                            channel_check_for_type(&temp.sid_data) == channel_type_escon) {
+                                channel[ESCON].list[channel[ESCON].count].devno = temp.devno; 
+                                channel[ESCON].count++; 
+
+                        }
+                }
+        }
+} 
+ 
+
+/*
+ * free specific channel from the channel[].list 
+ */  
+static int channel_free(int media, int devno)
+{
+        int     i;
+
+        for (i = 0; i < channel[media].count; i++) {       
+                if ((devno == channel[media].list[i].devno) &&
+                    ((channel[media].list[i].flag & CHANNEL_IN_USE) != 0x00)) {
+                        channel[media].list[i].flag &= ~CHANNEL_IN_USE;
+                        return 0;
+                }
+        }
+        printk(KERN_WARNING "channel: dev %04x is not a channel or in use\n", devno);
+        return -ENODEV; 
+}
+
+
+/*
+ * get specific channel from the channel[].list 
+ */  
+static int channel_get(int media, int devno)
+{
+        int     i;
+
+        for (i = 0; i < channel[media].count; i++) {       
+                if ((devno == channel[media].list[i].devno) &&
+                    ((channel[media].list[i].flag & CHANNEL_IN_USE) == 0x00)) {
+                        channel[media].list[i].flag |= CHANNEL_IN_USE;
+                        return channel[media].list[i].devno;             
+                }
+        }
+        printk(KERN_WARNING "channel: dev %04x is not a channel or in use\n", devno);
+        return -ENODEV; 
+
+}
+
+
+/*
+ * get the next free channel from the channel[].list 
+ */  
+static int channel_get_next(int media)
+{
+        int     i;
+
+        for (i = 0; i < channel[media].count; i++) {
+                if ((channel[media].list[i].flag & CHANNEL_IN_USE) == 0x00) {
+#ifdef DEBUG
+                        printk(KERN_DEBUG "channel: picked=%04x\n", channel[media].list[i].devno);
+#endif
+                        channel[media].list[i].flag |= CHANNEL_IN_USE;
+                        return channel[media].list[i].devno;        
+                }
+        }
+        return -ENODEV; 
+}
+ 
+
+/*
+ * picks the next free channel from the channel[].list 
+ */  
+static int channel_left(int media)
+{
+        return channel[media].left; 
+}
+
+
+/*
+ * defines all devices which are channels
+ */
+static channel_type_t channel_check_for_type (senseid_t *id)
+ {
+        channel_type_t type;
+
+        switch (id->cu_type) {
+                case 0x3088: 
+
+                        switch (id->cu_model) {
+                                case 0x08:    
+                                        type = channel_type_ctca;  /* 3088/08  ==> CTCA */
+                                        break; 
+
+                                case 0x1F:   
+                                        type = channel_type_escon; /* 3088/1F  ==> ESCON channel */
+                                        break;
+ 
+                                case 0x01:                         /* 3088/01  ==> P390 OSA emulation */
+                                case 0x60:                         /* 3088/60  ==> OSA/2 Adapter */
+                                        type = channel_type_unsupported;
+                                         break; 
+
+                                default:
+                                        type = channel_type_undefined;
+                                        printk(KERN_INFO "channel: Unknown model found 3088/%02x\n",id->cu_model);
+                        }
+                        break;
+
+                default:
+                        type = channel_type_none;
+
+        }
+        return type;
+}
+
+
+/*
+ *  sort the channel[].list
+ */
+static void channel_sort(struct devicelist list[], int n)
+{
+        int               i;
+        int               sorted = 0;
+        struct devicelist tmp;
+
+        while (!sorted) { 
+                sorted = 1;
+
+                for (i = 0; i < n-1; i++) {  
+                        if (list[i].devno > list[i+1].devno) {  
+                                tmp = list[i];
+                                list[i] = list[i+1];
+                                list[i+1] = tmp;
+                                sorted = 0;
+                        }
+                }
+        }
+} 
+
+
+/*
+ *   General routines 
+ *
+ */
+
+static int inline extract_channel_id(char *name)
+{
+        if (name[0] == 'c') 
+                return (name[3]-'0');
+        else
+                return (name[5]-'0');
+}
+
+
+static int inline extract_channel_media(char *name)
+{
+        if (name[0] == 'c') 
+                return CTC;
+        else
+                return ESCON;
+}
+
+
+static void ctc_tab_init(void)
+{                                    
+        int          m;
+        int          i;
+        static int   t;
+
+        if (t == 0){
+                for (m = 0; m < CHANNEL_MEDIA;  m++) {
+                        for (i = 0; i < MAX_ADAPTERS;  i++) {
+                                ctc_adapter[m][i].devno[WRITE] = -ENODEV;
+                                ctc_adapter[m][i].devno[READ] = -ENODEV;
+                        }
+                }
+                t = 1; 
+        }
+} 
+
+
+static int ctc_buffer_alloc(struct channel *ctc) {
+        
+        struct buffer    *p;
+        struct buffer    *q;
+
+        p = kmalloc(sizeof(p), GFP_KERNEL);
+        if (p == NULL) 
+                return -ENOMEM;
+        else {  
+                p->next = NULL;  
+                p->packets = 0;
+                p->block = (struct block *) __get_free_pages(GFP_KERNEL+GFP_DMA, 4);
+                if (p->block == NULL) {
+                        kfree(p);
+                        return -ENOMEM;
+                }
+        }
+   
+        if (ctc->free_anchor == NULL) 
+                ctc->free_anchor = p;  
+        else {  
+                 q = ctc->free_anchor;
+                 while (q->next != NULL) 
+                        q = q->next;
+                 q->next = p;
+        }
+        ctc->buffer_count++;   
+   return 0;
+}
+
+
+static int ctc_buffer_free(struct channel *ctc) {
+        
+        struct buffer    *p;
+
+           
+        if (ctc->free_anchor == NULL)
+                return -ENOMEM;
+        
+        p = ctc->free_anchor; 
+        ctc->free_anchor = p->next;
+        free_pages((__u32)p->block, 4);
+        kfree(p);
+
+        return 0;
+}
+
+
+static int inline ctc_buffer_swap(struct buffer **from, struct buffer **to) {
+        
+        struct buffer    *p = NULL;
+        struct buffer    *q = NULL;
+ 
+        if (*from == NULL)
+                return -ENOMEM; 
+
+        p = *from;
+        *from = p->next;
+        p->next = NULL;
+
+        if (*to == NULL)
+                *to = p;
+        else {
+                q = *to;
+                while (q->next != NULL)
+                        q = q->next;
+                q->next = p;
+
+        }
+        return 0;
+}
+
+
+/*
+ *   ctc_setup function 
+ *     this function is called for each ctc= keyword passed into the kernel 
+ *
+ *     valid parameter are: ctc=n,0xnnnn,0xnnnn,ctcx 
+ *     where n      is the channel protocol always 0 
+ *           0xnnnn is the cu number  read 
+ *           0xnnnn is the cu number  write 
+ *           ctcx can be ctc0 to ctc7 or escon0 to escon7 
+ */
+void ctc_setup(char *dev_name, int *ints)
+{
+        struct adapterlist tmp;
+
+        ctc_tab_init();
+        
+        ctc_no_auto = 1;
+
+        tmp.devno[WRITE] = -ENODEV;
+        tmp.devno[READ] = -ENODEV; 
+
+        switch (ints[0]) {
+              
+                case 3: /* write channel passed */
+                        tmp.devno[WRITE] = ints[3]; 
+                        
+                case 2: /* read channel passed */
+                        tmp.devno[READ] = ints[2];
+                        if (tmp.devno[WRITE] == -ENODEV)
+                                tmp.devno[WRITE] = tmp.devno[READ]++; 
+
+                case 1: /* protocol type passed */
+                        tmp.protocol    = ints[1];
+                        if (tmp.protocol == 0) {
+                                break;    
+                        } else {
+                                printk(KERN_WARNING "%s: wrong Channel protocol type passed\n", dev_name);
+                                return;
+                        }
+
+                default: 
+                        printk(KERN_WARNING "%s: wrong number of parameter passed\n", dev_name);
+                        return;
+        }
+        ctc_adapter[extract_channel_media(dev_name)][extract_channel_id(dev_name)] = tmp; 
+#ifdef DEBUG
+        printk(DEBUG "%s: protocol=%x read=%04x write=%04x\n",
+             dev_name, tmp.protocol, tmp.devno[READ], tmp.devno[WRITE]);
+#endif  
+        return;
+        
+}
+
+
+/*
+ *   ctc_probe 
+ *      this function is called for each channel network device, 
+ *      which is defined in the /init/main.c 
+ */
+int ctc_probe(struct device *dev)
+{       
+        int                rc;
+        int                c;
+        int                i;
+        int                m;
+
+        struct ctc_priv    *privptr;
+
+        /* Only the first time the ctc_probe gets control */
+        if (channel_tab_initialized == 0) {  
+                channel_init();  
+
+
+        }
+        
+        ctc_tab_init();
+
+        m = extract_channel_media(dev->name);
+        i = extract_channel_id(dev->name);
+        
+        if (channel_left(m) <=1) 
+                return -ENODEV;
+
+        dev->priv = kmalloc(sizeof(struct ctc_priv), GFP_KERNEL);
+        if (dev->priv == NULL)
+                return -ENOMEM;
+        memset(dev->priv, 0, sizeof(struct ctc_priv));  
+        privptr = (struct ctc_priv *) (dev->priv);
+
+        
+        for (c = 0; c < 2; c++) {
+                privptr->channel[c].devstat = kmalloc(sizeof(devstat_t), GFP_KERNEL);
+                if (privptr->channel[c].devstat == NULL){
+                        if (i == WRITE)
+                                kfree(privptr->channel[READ].devstat);
+                        return -ENOMEM;  
+                }
+                memset(privptr->channel[c].devstat, 0, sizeof(devstat_t));
+
+                if (ctc_no_auto == 0) 
+                         ctc_adapter[m][i].devno[c] = channel_get_next(m);
+                else 
+                         ctc_adapter[m][i].devno[c] = channel_get(m, ctc_adapter[m][i].devno[c]);
+
+                if ( ctc_adapter[m][i].devno[c] != -ENODEV){
+                         rc = request_irq(get_irq_by_devno(ctc_adapter[m][i].devno[c]),
+                              (void *)ctc_irq_handler, SA_INTERRUPT, dev->name, 
+                              privptr->channel[c].devstat);
+                         if (rc) { 
+                                 printk(KERN_WARNING "%s: requested device busy %02x\n", dev->name, rc);
+                                 return -EBUSY;
+                         }
+                } else {        
+                        if (i == WRITE) {
+                                free_irq(get_irq_by_devno(ctc_adapter[m][i].devno[c]), privptr->channel[i].devstat);
+                                channel_free(m, ctc_adapter[m][i].devno[READ]);
+                                kfree(privptr->channel[READ].devstat);
+                        } 
+                        kfree(privptr->channel[i].devstat);
+                        return -ENODEV;
+                }
+        }
+
+        privptr->channel[READ].devno = ctc_adapter[m][i].devno[READ];
+        privptr->channel[READ].irq = get_irq_by_devno(ctc_adapter[m][i].devno[READ]);
+        privptr->channel[WRITE].devno = ctc_adapter[m][i].devno[WRITE];
+        privptr->channel[WRITE].irq = get_irq_by_devno(ctc_adapter[m][i].devno[WRITE]);
+        privptr->protocol = ctc_adapter[m][i].protocol;
+        channel[m].left = channel[m].left - 2;
+
+        printk(KERN_INFO "%s: read dev: %04x irq: %04x - write dev: %04x irq: %04x \n",
+            dev->name, privptr->channel[READ].devno,   privptr->channel[READ].irq,
+            privptr->channel[WRITE].devno,  privptr->channel[WRITE].irq); 
+
+        dev->mtu             = CTC_DEFAULT_MTU_SIZE;
+        dev->hard_start_xmit = ctc_tx;
+        dev->open            = ctc_open;
+        dev->stop            = ctc_release;
+        dev->get_stats       = ctc_stats;
+        dev->change_mtu      = ctc_change_mtu;
+        dev->hard_header_len = 0;
+        dev->addr_len        = 0;
+        dev->type            = ARPHRD_SLIP;
+        dev->tx_queue_len    = 100;
+        dev_init_buffers(dev);
+        dev->flags           = IFF_POINTOPOINT | IFF_NOARP;   
+
+        return 0;
+} 
+
+
+/*
+ *   Interrupt processing 
+ *
+ */
+
+static void inline ccw_check_return_code (struct device *dev, int return_code)
+{
+        if (return_code != 0) {
+                switch (return_code) {
+                        case -EBUSY:  
+                                printk(KERN_INFO "%s: Busy !\n", dev->name);
+                                break;
+                        case -ENODEV:
+                                printk(KERN_EMERG "%s: Invalid device called for IO\n", dev->name);
+                                break;
+                        case -EIO:
+                                printk(KERN_EMERG "%s: Status pending... \n", dev->name);
+                                break;
+                        default:
+                                printk(KERN_EMERG "%s: Unknown error in Do_IO %04x\n", 
+                                    dev->name, return_code);
+                }
+        }
+} 
+
+
+static void inline ccw_check_unit_check (struct device *dev, char sense)
+{
+#ifdef DEBUG
+        printk(KERN_INFO "%s: Unit Check with sense code: %02x\n",
+            dev->name, sense);
+#endif
+
+        if (sense & 0x40) {
+#ifdef DEBUG
+                if (sense & 0x01) 
+                        printk(KERN_DEBUG "%s: Interface disconnect or Selective reset occurred (remote side)\n", dev->name);
+                else 
+                        printk(KERN_DEBUG "%s: System reset occured (remote side)\n", dev->name);
+#endif
+        } else if (sense & 0x20) {
+                if (sense & 0x04)
+                        printk(KERN_WARNING "%s: Data-streaming timeout)\n", dev->name);
+                else 
+                        printk(KERN_WARNING "%s: Data-transfer parity error\n", dev->name);
+        } else if (sense & 0x10) {
+                if (sense & 0x20)
+                        printk(KERN_WARNING "%s: Hardware malfunction (remote side)\n", dev->name);
+                else 
+                        printk(KERN_WARNING "%s: Read-data parity error (remote side)\n", dev->name);
+        }
+
+} 
+
+
+static void ctc_irq_handler (int irq, void *initparm, struct pt_regs *regs)
+{
+        int               rc = 0;
+        __u32             parm;
+        __u8              flags = 0x00;
+        struct  channel   *ctc = NULL;
+        struct  ctc_priv  *privptr = NULL;
+        struct  device    *dev = NULL;    
+        
+        ccw1_t            ccw_set_x_mode[2] = {{CCW_CMD_SET_EXTENDED, CCW_FLAG_SLI | CCW_FLAG_CC, 0, NULL},
+                                               {CCW_CMD_NOOP, CCW_FLAG_SLI, 0, NULL}}; 
+
+        devstat_t *devstat = ((devstat_t *)initparm);
+
+        /* Bypass all 'unsolited interrupts' */
+        if (devstat->intparm == 0) {
+#ifdef DEBUG
+                printk(KERN_DEBUG "ctc: unsolited interrupt for device: %04x received c-%02x d-%02x f-%02x\n",
+                    devstat->devno, devstat->cstat, devstat->dstat, devstat->flag);
+#endif 
+                /* FIXME - find the related intparm!!! No IO outstanding!!!! */
+                return;
+        }
+
+        ctc = (struct channel *) (devstat->intparm);
+        dev = (struct device *) ctc->dev;
+        privptr = dev->priv;
+
+#ifdef DEBUG
+        printk(KERN_DEBUG "%s: interrupt for device: %04x received c-%02x d-%02x f-%02x state-%02x\n",
+            dev->name, ctc->devno, devstat->cstat, devstat->dstat, devstat->flag, ctc->state);
+#endif 
+
+        /* Check for good subchannel return code, otherwise error message */
+        if (devstat->cstat) {
+                printk(KERN_WARNING "%s: subchannel check for device: %04x - %02x\n", 
+                    dev->name, ctc->devno, devstat->cstat);
+                return;
+        }
+
+
+        /* Check the reason-code of a unit check */
+        if (devstat->dstat & DEV_STAT_UNIT_CHECK)
+                ccw_check_unit_check(dev, devstat->ii.sense.data[0]);
+
+
+        /* State machine to bring the connection up / down and to restart */ 
+
+        ctc->last_dstat = devstat->dstat;
+
+        switch (ctc->state) {   
+
+                case CTC_STOP:           /* HALT_IO issued by ctc_release (halt sequence) */
+                        if (!devstat->flag & DEVSTAT_FINAL_STATUS)
+                                return;
+                        wake_up(&ctc->wait);   /* wake up ctc_release */
+                        return;
+ 
+
+                case CTC_START_HALT_IO:  /* HALT_IO issued by ctc_open (start sequence) */
+                        if (!devstat->flag & DEVSTAT_FINAL_STATUS)
+                                return;
+
+                        ctc->state = CTC_START_SET_X_MODE;
+                        parm = (__u32) ctc;
+                        rc = do_IO (ctc->irq, &ccw_set_x_mode[0], parm, 0xff, flags);
+                        if (rc != 0)
+                                ccw_check_return_code(dev, rc);
+                        return;
+ 
+        
+                case CTC_START_SET_X_MODE:
+                        if (devstat->dstat & DEV_STAT_UNIT_CHECK) {
+                                if ((devstat->ii.sense.data[0] & 0x41) != 0x41 ||
+                                    (devstat->ii.sense.data[0] & 0x40) != 0x40) {
+                                        wake_up(&ctc->wait);  /* wake up ctc_open (READ or WRITE) */
+                                        return; 
+                                }        
+                        }
+                        if (!devstat->flag & DEVSTAT_FINAL_STATUS)
+                                return;
+                        ctc->state =  CTC_START_SELECT;
+ 
+
+                case CTC_START_SELECT:
+                        if (!ctc->flag & CTC_WRITE) {
+                                ctc->state = CTC_START_READ_TEST;
+                                ctc->ccw[1].cda  = (char *)virt_to_phys(ctc->free_anchor->block);
+                                parm = (__u32) ctc;
+                                rc = do_IO (ctc->irq, &ctc->ccw[0], parm, 0xff, flags );
+                                if (rc != 0) 
+                                        ccw_check_return_code(dev, rc);
+                                wake_up(&ctc->wait);  /* wake up ctc_open (READ) */
+
+                        } else {
+                                ctc->state = CTC_START_WRITE_TEST;
+                                /* ADD HERE THE RIGHT PACKET TO ISSUE A ROUND TRIP - PART 1 */
+                                ctc->ccw[1].count = 0;
+                                ctc->ccw[1].cda  = (char *)virt_to_phys(ctc->free_anchor->block);
+                                parm = (__u32) ctc; 
+                                rc = do_IO (ctc->irq, &ctc->ccw[0], parm, 0xff, flags);
+                                if (rc != 0)
+                                        ccw_check_return_code(dev, rc);
+                        }
+                        return;
+
+
+                case CTC_START_READ_TEST:
+                        if (devstat->dstat & DEV_STAT_UNIT_CHECK) {
+                                if ((devstat->ii.sense.data[0] & 0x41) == 0x41 ||
+                                    (devstat->ii.sense.data[0] & 0x40) == 0x40 ||
+                                    devstat->ii.sense.data[0] == 0                ) {
+                                        init_timer(&ctc->timer);
+                                        ctc->timer.function = (void *)ctc_read_retry; 
+                                        ctc->timer.data = (__u32)ctc;
+                                        ctc->timer.expires = jiffies + 10*HZ;
+                                        add_timer(&ctc->timer);
+#ifdef DEBUG 
+                                        printk(KERN_DEBUG "%s: read connection restarted\n",dev->name); 
+#endif
+                                }
+                                return;
+                        }
+
+                        if ((devstat->dstat &  ~(DEV_STAT_CHN_END | DEV_STAT_DEV_END)) != 0x00) {
+                                if ((devstat->dstat & DEV_STAT_ATTENTION) && 
+                                    (devstat->dstat & DEV_STAT_BUSY)) {
+                                        printk(KERN_WARNING "%s: read channel is connected with the remote side read channel\n", dev->name);
+                                } 
+                                wake_up(&privptr->channel[WRITE].wait);  /* wake up ctc_open (WRITE) */
+                                return;
+                        }
+
+                        ctc->state = CTC_START_READ;
+                        set_bit(0, (void *)&ctc->IO_active);
+
+                        /* ADD HERE THE RIGHT PACKET TO ISSUE A ROUND TRIP - PART 2 */
+                        /* wake_up(&privptr->channel[WRITE].wait);*/  /* wake up ctc_open (WRITE) */
+
+
+                case CTC_START_READ: 
+                        if (devstat->dstat & DEV_STAT_UNIT_CHECK) {
+                                if ((devstat->ii.sense.data[0] & 0x41) == 0x41 ||
+                                    (devstat->ii.sense.data[0] & 0x40) == 0x40 ||
+                                    devstat->ii.sense.data[0] == 0               ) {
+                                        privptr->stats.rx_errors++;
+                                        set_bit(TB_RETRY, (void *)&dev->tbusy);
+                                        init_timer(&ctc->timer);
+                                        ctc->timer.function = (void *)ctc_read_retry; 
+                                        ctc->timer.data = (__u32)ctc;
+                                        ctc->timer.expires = jiffies + 30*HZ;
+                                        add_timer(&ctc->timer); 
+                                        printk(KERN_INFO "%s: connection restarted!! problem on remote side\n",dev->name);
+                                }
+                                return;
+                        }
+
+                        if(!devstat->flag & DEVSTAT_FINAL_STATUS)
+                                return; 
+
+                        clear_bit(TB_RETRY, (void *)&dev->tbusy);
+                        
+                        ctc_buffer_swap(&ctc->free_anchor, &ctc->proc_anchor);
+
+                        if (ctc->free_anchor != NULL) {  
+                                ctc->ccw[1].cda  = (char *)virt_to_phys(ctc->free_anchor->block);
+                                parm = (__u32) ctc;
+                                rc = do_IO (ctc->irq, &ctc->ccw[0], parm, 0xff, flags );
+                                if (rc != 0) 
+                                        ccw_check_return_code(dev, rc);
+                        } else {
+                                clear_bit(0, (void *)&ctc->IO_active);    
+#ifdef DEBUG
+                                printk(KERN_DEBUG "%s: No HOT READ started in IRQ\n",dev->name);
+#endif
+                        }
+                        
+                        if (test_and_set_bit(CTC_BH_ACTIVE, (void *)&ctc->flag_a) == 0) {
+                                queue_task(&ctc->tq, &tq_immediate);
+                                mark_bh(IMMEDIATE_BH);
+                        }
+                        return;
+
+
+                case CTC_START_WRITE_TEST:
+                        if (devstat->dstat & DEV_STAT_UNIT_CHECK) {
+                                if ((devstat->ii.sense.data[0] & 0x41) == 0x41 ||
+                                    (devstat->ii.sense.data[0] & 0x40) == 0x40 ||
+                                    devstat->ii.sense.data[0] == 0                ) {
+                                        init_timer(&ctc->timer);
+                                        ctc->timer.function = (void *)ctc_write_retry; 
+                                        ctc->timer.data = (__u32)ctc;
+                                        ctc->timer.expires = jiffies + 10*HZ;
+                                        add_timer(&ctc->timer);
+#ifdef DEBUG
+                                        printk(KERN_DEBUG "%s: write connection restarted\n",dev->name);
+#endif
+                                }
+                                return;
+                        }
+
+                        ctc->state = CTC_START_WRITE;
+                        wake_up(&ctc->wait);  /* wake up ctc_open (WRITE) */
+                        return;
+ 
+
+                case CTC_START_WRITE:
+                        if (devstat->dstat & DEV_STAT_UNIT_CHECK) {
+                                privptr->stats.tx_errors += ctc->proc_anchor->packets;
+#ifdef DEBUG
+                                printk(KERN_DEBUG "%s: Unit Check on write channel\n",dev->name);
+#endif
+                        } else { 
+                                if (!devstat->flag & DEVSTAT_FINAL_STATUS)
+                                        return; 
+                                privptr->stats.tx_packets += ctc->proc_anchor->packets;
+                        } 
+
+                        ctc->proc_anchor->block->length = 0;
+                        ctc_buffer_swap(&ctc->proc_anchor, &ctc->free_anchor);
+                        clear_bit(TB_NOBUFFER, (void *)&dev->tbusy);      
+
+                        if (ctc->proc_anchor != NULL) {  
+#ifdef DEBUG
+                                printk(KERN_DEBUG "%s: IRQ early swap buffer\n",dev->name); 
+#endif
+                                ctc->ccw[1].count = ctc->proc_anchor->block->length;
+                                ctc->ccw[1].cda  = (char *)virt_to_phys(ctc->proc_anchor->block);
+                                parm = (__u32) ctc;
+                                rc = do_IO (ctc->irq, &ctc->ccw[0], parm, 0xff, flags );
+                                if (rc != 0) 
+                                        ccw_check_return_code(dev, rc);
+                                dev->trans_start = jiffies;
+                                return;
+
+                        }
+
+                        if (ctc->free_anchor->block->length != 0) {
+                                if (test_and_set_bit(TB_TX, (void *)&dev->tbusy) == 0) {     /* set transmission to busy */
+                                        ctc_buffer_swap(&ctc->free_anchor, &ctc->proc_anchor);
+                                        clear_bit(TB_TX, (void *)&dev->tbusy);
+#ifdef DEBUG
+                                        printk(KERN_DEBUG "%s: last buffer move in IRQ\n",dev->name); 
+#endif
+                                        ctc->ccw[1].count = ctc->proc_anchor->block->length;
+                                        ctc->ccw[1].cda  = (char *)virt_to_phys(ctc->proc_anchor->block);
+                                        parm = (__u32) ctc;
+                                        rc = do_IO (ctc->irq, &ctc->ccw[0], parm, 0xff, flags );
+                                        if (rc != 0) 
+                                                ccw_check_return_code(dev, rc);
+                                        dev->trans_start = jiffies;
+                                        return;
+                                }
+                        } 
+
+                        clear_bit(0, (void *)&ctc->IO_active);              /* set by ctc_tx or ctc_bh */
+                        return;
+
+ 
+                default: 
+                        printk(KERN_WARNING "%s: wrong selection code - irq\n",dev->name);
+                        return;  
+        }
+} 
+
+
+static void ctc_irq_bh (struct channel *ctc)
+{
+        int                rc = 0;
+        __u16              data_len;
+        __u32              parm;
+
+        __u8               flags = 0x00;
+        __u32              saveflags;
+        struct device      *dev;
+        struct ctc_priv    *privptr;             
+        struct packet      *lp;
+        struct sk_buff     *skb;
+
+        dev = (struct device *) ctc->dev; 
+        privptr = (struct ctc_priv *) dev->priv; 
+   
+#ifdef DEBUG
+        printk(KERN_DEBUG "%s: bh routine - state-%02x\n" ,dev->name, ctc->state);
+#endif 
+
+        while (ctc->proc_anchor != NULL) { 
+
+                lp = &ctc->proc_anchor->block->data;
+
+                while ((__u8 *) lp < (__u8 *) &ctc->proc_anchor->block->length + ctc->proc_anchor->block->length) {
+                        data_len = lp->length - PACKET_HEADER_LENGTH;
+                        skb = dev_alloc_skb(data_len); 
+                        if (skb) { 
+                                memcpy(skb_put(skb, data_len),&lp->data, data_len);
+                                skb->mac.raw = skb->data;
+                                skb->dev = dev;
+                                skb->protocol = htons(ETH_P_IP);
+                                skb->ip_summed = CHECKSUM_UNNECESSARY; /* no UC happend!!! */
+                                netif_rx(skb);
+                                privptr->stats.rx_packets++;
+                        } else {
+                                privptr->stats.rx_dropped++; 
+                                printk(KERN_WARNING "%s: is low on memory\n",dev->name);
+                        }
+                        (__u8 *)lp += lp->length;
+                }
+
+                s390irq_spin_lock_irqsave(ctc->irq, saveflags);
+                ctc_buffer_swap(&ctc->proc_anchor, &ctc->free_anchor);
+
+                if (test_and_set_bit(0, (void *)&ctc->IO_active) == 0) {
+#ifdef DEBUG
+                        printk(KERN_DEBUG "%s: HOT READ started in bh routine\n" ,dev->name);
+#endif 
+                        ctc->ccw[1].cda  = (char *)virt_to_phys(ctc->free_anchor->block);
+                        parm = (__u32) ctc; 
+                        rc = do_IO (ctc->irq, &ctc->ccw[0], parm, 0xff, flags );
+                        if (rc != 0) 
+                                 ccw_check_return_code(dev, rc);
+                } 
+                s390irq_spin_unlock_irqrestore(ctc->irq, saveflags); 
+        }
+        clear_bit(CTC_BH_ACTIVE, (void *)&ctc->flag_a);
+        return;
+}  
+
+
+static void ctc_read_retry (struct channel *ctc)
+{
+        int                rc = 0;
+        __u32              parm;
+        __u8               flags = 0x00;
+        __u32              saveflags;
+        struct device      *dev;
+
+        dev = (struct device *) ctc->dev; 
+   
+#ifdef DEBUG
+        printk(KERN_DEBUG "%s: read retry - state-%02x\n" ,dev->name, ctc->state);
+#endif 
+        s390irq_spin_lock_irqsave(ctc->irq, saveflags);
+        ctc->ccw[1].cda  = (char *)virt_to_phys(ctc->free_anchor->block);
+        parm = (__u32) ctc; 
+        rc = do_IO (ctc->irq, &ctc->ccw[0], parm, 0xff, flags );
+        s390irq_spin_unlock_irqrestore(ctc->irq, saveflags);
+        if (rc != 0) 
+                ccw_check_return_code(dev, rc);
+        return;
+}  
+ 
+
+static void ctc_write_retry (struct channel *ctc)
+{
+        int                rc = 0;
+        __u32              parm;
+        __u8               flags = 0x00;
+        __u32              saveflags;
+        struct device      *dev;
+
+        dev = (struct device *) ctc->dev; 
+   
+#ifdef DEBUG
+        printk(KERN_DEBUG "%s: write retry - state-%02x\n" ,dev->name, ctc->state);
+#endif 
+        s390irq_spin_lock_irqsave(ctc->irq, saveflags);
+        ctc->ccw[1].count = 0;
+        ctc->ccw[1].cda  = (char *)virt_to_phys(ctc->proc_anchor->block);
+        parm = (__u32) ctc; 
+        rc = do_IO (ctc->irq, &ctc->ccw[0], parm, 0xff, flags );
+        s390irq_spin_unlock_irqrestore(ctc->irq, saveflags);
+        if (rc != 0) 
+                ccw_check_return_code(dev, rc);
+        return;
+}  
+
+
+
+/*
+ *   ctc_open
+ *
+ */
+static int ctc_open(struct device *dev)
+{
+        int                rc;
+        int                i;
+        int                j;
+        __u8               flags = 0x00;
+        __u32              saveflags;
+        __u32              parm;
+        struct ctc_priv    *privptr;
+        struct wait_queue  wait = { current, NULL };
+        struct timer_list  timer;
+
+
+        dev->tbusy = 1;
+        dev->start = 0;
+
+        privptr = (struct ctc_priv *) (dev->priv);
+        
+        privptr->channel[READ].flag  = 0x00;
+        privptr->channel[WRITE].flag = CTC_WRITE;
+
+        for (i = 0; i < 2;  i++) { 
+                for (j = 0; j < CTC_BLOCKS;  j++) { 
+                        rc = ctc_buffer_alloc(&privptr->channel[i]);
+                        if (rc != 0)
+                                return -ENOMEM;
+                }
+                privptr->channel[i].tq.next = NULL;
+                privptr->channel[i].tq.sync = 0;
+                privptr->channel[i].tq.routine = (void *)(void *)ctc_irq_bh;
+                privptr->channel[i].tq.data = &privptr->channel[i]; 
+
+                privptr->channel[i].dev = dev;
+                
+                privptr->channel[i].flag_a = 0;
+
+                privptr->channel[i].ccw[0].cmd_code  = CCW_CMD_PREPARE;
+                privptr->channel[i].ccw[0].flags     = CCW_FLAG_SLI | CCW_FLAG_CC;
+                privptr->channel[i].ccw[0].count     = 0;
+                privptr->channel[i].ccw[0].cda       = NULL;
+                if (i == READ) {  
+                        privptr->channel[i].ccw[1].cmd_code  = CCW_CMD_READ;
+                        privptr->channel[i].ccw[1].flags     = CCW_FLAG_SLI;
+                        privptr->channel[i].ccw[1].count     = 0xffff;   /* MAX size */
+                        privptr->channel[i].ccw[1].cda       = NULL;
+                } else {
+                        privptr->channel[i].ccw[1].cmd_code = CCW_CMD_WRITE;
+                        privptr->channel[i].ccw[1].flags    = CCW_FLAG_SLI | CCW_FLAG_CC;
+                        privptr->channel[i].ccw[1].count    = 0;
+                        privptr->channel[i].ccw[1].cda      = NULL;
+                }
+                privptr->channel[i].ccw[2].cmd_code = CCW_CMD_NOOP;      /* jointed CE+DE */
+                privptr->channel[i].ccw[2].flags    = CCW_FLAG_SLI;
+                privptr->channel[i].ccw[2].count    = 0;
+                privptr->channel[i].ccw[2].cda      = NULL;
+                
+                privptr->channel[i].flag  &= ~CTC_TIMER;
+                init_timer(&timer);
+                timer.function = (void *)ctc_timer; 
+                timer.data = (__u32)&privptr->channel[i];
+                timer.expires = jiffies + 150*HZ;                        /* time to connect with the remote side */
+                add_timer(&timer);
+
+                s390irq_spin_lock_irqsave(privptr->channel[i].irq, saveflags);
+                parm = (unsigned long) &privptr->channel[i]; 
+                privptr->channel[i].state = CTC_START_HALT_IO;
+                rc = halt_IO(privptr->channel[i].irq, parm, flags);
+                add_wait_queue(&privptr->channel[i].wait, &wait);
+                current->state = TASK_INTERRUPTIBLE;
+                s390irq_spin_unlock_irqrestore(privptr->channel[i].irq, saveflags);
+                schedule();
+                remove_wait_queue(&privptr->channel[i].wait, &wait);
+                if(rc != 0)
+                        ccw_check_return_code(dev, rc);
+                if((privptr->channel[i].flag & CTC_TIMER) == 0x00)
+                        del_timer(&timer);
+        }
+
+        if ((((privptr->channel[READ].last_dstat | privptr->channel[WRITE].last_dstat) & 
+               ~(DEV_STAT_CHN_END | DEV_STAT_DEV_END)) != 0x00) ||
+            (((privptr->channel[READ].flag | privptr->channel[WRITE].flag) & CTC_TIMER) != 0x00)) {
+#ifdef DEBUG
+                printk(KERN_DEBUG "%s: channel problems during open - read: %02x -  write: %02x\n",
+                    dev->name, privptr->channel[READ].last_dstat, privptr->channel[WRITE].last_dstat);
+#endif 
+                printk(KERN_INFO "%s: remote side is currently not ready\n", dev->name);
+                
+                for (i = 0; i < 2;  i++) {
+                        s390irq_spin_lock_irqsave(privptr->channel[i].irq, saveflags);
+                        parm = (unsigned long) &privptr->channel[i];
+                        privptr->channel[i].state = CTC_STOP;
+                        rc = halt_IO(privptr->channel[i].irq, parm, flags);
+                        s390irq_spin_unlock_irqrestore(privptr->channel[i].irq, saveflags);
+                        if (rc != 0)
+                                ccw_check_return_code(dev, rc);
+                        for (j = 0; j < CTC_BLOCKS;  j++) 
+                                ctc_buffer_free(&privptr->channel[i]);
+                }
+                return -EIO;
+        }
+
+        printk(KERN_INFO "%s: connected with remote side\n",dev->name);
+        dev->start = 1;
+        dev->tbusy = 0;
+        return 0;
+}
+
+
+static void ctc_timer (struct channel *ctc)
+{
+#ifdef DEBUG
+        struct device      *dev;
+
+        dev = (struct device *) ctc->dev; 
+        printk(KERN_DEBUG "%s: timer return\n" ,dev->name);
+#endif
+        ctc->flag |= CTC_TIMER;
+        wake_up(&ctc->wait);  
+        return;
+}  
+
+/*
+ *   ctc_release 
+ *
+ */
+static int ctc_release(struct device *dev)
+{   
+        int                rc;
+        int                i;
+        int                j;
+        __u8               flags = 0x00;
+        __u32              saveflags;
+        __u32              parm;
+        struct ctc_priv    *privptr;
+        struct wait_queue  wait = { current, NULL }; 
+
+        privptr = (struct ctc_priv *) dev->priv;  
+   
+        dev->start = 0;
+        set_bit(TB_STOP, (void *)&dev->tbusy);    
+        
+        for (i = 0; i < 2;  i++) {
+                s390irq_spin_lock_irqsave(privptr->channel[i].irq, saveflags);
+                privptr->channel[i].state = CTC_STOP;
+                parm = (__u32) &privptr->channel[i]; 
+                rc = halt_IO (privptr->channel[i].irq, parm, flags );
+                add_wait_queue(&privptr->channel[i].wait, &wait);
+                current->state = TASK_INTERRUPTIBLE;
+                s390irq_spin_unlock_irqrestore(privptr->channel[i].irq, saveflags); 
+                schedule();
+                remove_wait_queue(&privptr->channel[i].wait, &wait);
+                if (rc != 0) {
+                        ccw_check_return_code(dev, rc);
+                }
+                 
+                for (j = 0; j < CTC_BLOCKS;  j++) {  
+                        ctc_buffer_swap(&privptr->channel[i].proc_anchor, &privptr->channel[i].free_anchor);
+                        ctc_buffer_free(&privptr->channel[i]); 
+                }
+        }
+
+        if (((privptr->channel[READ].last_dstat | privptr->channel[WRITE].last_dstat) &
+            ~(DEV_STAT_CHN_END | DEV_STAT_DEV_END)) != 0x00) {
+                printk(KERN_WARNING "%s: channel problems during close - read: %02x -  write: %02x\n",
+                    dev->name, privptr->channel[READ].last_dstat, privptr->channel[WRITE].last_dstat);
+                return -EIO;
+        }
+
+        return 0;
+}  
+
+
+/*
+ *   ctc_tx 
+ *
+ *
+ */
+static int ctc_tx(struct sk_buff *skb, struct device *dev)
+{
+        int                rc;
+        __u32              parm;
+        __u8               flags = 0x00;
+        __u32              saveflags;
+        struct ctc_priv    *privptr;
+        struct packet      *lp;
+   
+        privptr = (struct ctc_priv *) (dev->priv);
+
+        if (skb == NULL) { 
+                printk(KERN_WARNING "%s: NULL pointer as sk_buffer passed\n", dev->name);
+                privptr->stats.tx_dropped++;
+                return -EIO;
+        }
+        
+        if (dev->tbusy != 0) {
+                return -EBUSY;
+        } 
+
+        if (test_and_set_bit(TB_TX, (void *)&dev->tbusy) != 0) {                /* set transmission to busy */
+                return -EBUSY;
+        } 
+
+        if (65535 - privptr->channel[WRITE].free_anchor->block->length - PACKET_HEADER_LENGTH <= skb->len + PACKET_HEADER_LENGTH + 2) {
+#ifdef DEBUG
+                printk(KERN_DEBUG "%s: early swap\n", dev->name);
+#endif
+                s390irq_spin_lock_irqsave(privptr->channel[WRITE].irq, saveflags);
+                ctc_buffer_swap(&privptr->channel[WRITE].free_anchor, &privptr->channel[WRITE].proc_anchor);
+                s390irq_spin_unlock_irqrestore(privptr->channel[WRITE].irq, saveflags);
+                if (privptr->channel[WRITE].free_anchor == NULL){
+                        set_bit(TB_NOBUFFER, (void *)&dev->tbusy);
+                        clear_bit(TB_TX, (void *)&dev->tbusy);
+                        return -EBUSY;
+                }
+        }
+        
+        if (privptr->channel[WRITE].free_anchor->block->length == 0) {
+                privptr->channel[WRITE].free_anchor->block->length = BLOCK_HEADER_LENGTH; 
+                privptr->channel[WRITE].free_anchor->packets = 0;
+        } 
+
+
+        (__u8 *)lp = (__u8 *) &privptr->channel[WRITE].free_anchor->block->length + privptr->channel[WRITE].free_anchor->block->length;
+        privptr->channel[WRITE].free_anchor->block->length += skb->len + PACKET_HEADER_LENGTH;
+        lp->length = skb->len + PACKET_HEADER_LENGTH; 
+        lp->type = 0x0800; 
+        lp->unused = 0;
+        memcpy(&lp->data, skb->data, skb->len); 
+        (__u8 *) lp += lp->length; 
+        lp->length = 0;
+        dev_kfree_skb(skb);
+        privptr->channel[WRITE].free_anchor->packets++;
+
+        if (test_and_set_bit(0, (void *)&privptr->channel[WRITE].IO_active) == 0) {
+                s390irq_spin_lock_irqsave(privptr->channel[WRITE].irq, saveflags);
+                ctc_buffer_swap(&privptr->channel[WRITE].free_anchor,&privptr->channel[WRITE].proc_anchor); 
+                privptr->channel[WRITE].ccw[1].count = privptr->channel[WRITE].proc_anchor->block->length;
+                privptr->channel[WRITE].ccw[1].cda   = (char *)virt_to_phys(privptr->channel[WRITE].proc_anchor->block);
+                parm = (__u32) &privptr->channel[WRITE];  
+                rc = do_IO (privptr->channel[WRITE].irq, &privptr->channel[WRITE].ccw[0], parm, 0xff, flags );
+                if (rc != 0) 
+                        ccw_check_return_code(dev, rc);
+                dev->trans_start = jiffies;
+                s390irq_spin_unlock_irqrestore(privptr->channel[WRITE].irq, saveflags);
+        }
+
+        if (privptr->channel[WRITE].free_anchor == NULL)
+                set_bit(TB_NOBUFFER, (void *)&dev->tbusy);
+
+        clear_bit(TB_TX, (void *)&dev->tbusy);
+        return 0;
+} 
+
+
+/*
+ *   ctc_change_mtu 
+ *
+ *   S/390 can handle MTU sizes from 576 to 32760 for VM, VSE
+ *                                   576 to 65527 for OS/390
+ *
+ */
+static int ctc_change_mtu(struct device *dev, int new_mtu)
+{
+        if ((new_mtu < 576) || (new_mtu > 65528))
+                return -EINVAL;
+        dev->mtu = new_mtu;
+        return 0;
+}
+
+
+/*
+ *   ctc_stats
+ *
+ */
+struct net_device_stats *ctc_stats(struct device *dev)
+{
+         struct ctc_priv *privptr;
+   
+         privptr = dev->priv;
+         return &privptr->stats;
+}  
+
+
+/* Module code goes here */
+
+/*
+                free_irq(privptr->channel[i].irq, privptr->channel[i].devstat);
+                kfree(privptr->channel[i].devstat); 
+
+*/
+/* --- This is the END my friend --- */

FUNET's LINUX-ADM group, linux-adm@nic.funet.fi
TCL-scripts by Sam Shen (who was at: slshen@lbl.gov)