patch-2.4.2 linux/drivers/s390/char/tape.c

Next file: linux/drivers/s390/char/tape.h
Previous file: linux/drivers/s390/char/hwc_tty.c
Back to the patch index
Back to the overall index

diff -u --recursive --new-file v2.4.1/linux/drivers/s390/char/tape.c linux/drivers/s390/char/tape.c
@@ -0,0 +1,723 @@
+
+/***********************************************************************
+ *  drivers/s390/char/tape.c
+ *    tape device driver for S/390 tapes.
+ *
+ *  S390 version
+ *    Copyright (C) 2000 IBM Corporation
+ *    Author(s): Tuan Ngo-Anh <ngoanh@de.ibm.com>
+ *               Carsten Otte <cotte@de.ibm.com>
+ *
+ *  UNDER CONSTRUCTION: Work in progress... :-)
+ ***********************************************************************
+ */
+
+#include "tapedefs.h"
+
+#include <linux/config.h>
+#include <linux/stddef.h>
+#include <linux/kernel.h>
+#include <linux/version.h>
+#include <linux/proc_fs.h>
+#include <asm/types.h>
+#include <asm/ccwcache.h>
+#include <asm/idals.h>
+#include <asm/ebcdic.h>
+#include <linux/compatmac.h>
+#ifdef MODULE
+#include <linux/module.h>
+#endif   
+#ifdef TAPE_DEBUG
+#include <asm/debug.h>
+#endif
+#ifdef CONFIG_S390_TAPE_DYNAMIC
+#include <asm/s390dyn.h>
+#endif
+#include "tape.h"
+#include "tape3490.h"
+#include "tape3480.h"
+
+#define PRINTK_HEADER "T390:"
+
+/* state handling routines */
+inline void tapestate_set (tape_info_t * tape, int newstate);
+inline int tapestate_get (tape_info_t * tape);
+void tapestate_event (tape_info_t * tape, int event);
+
+/* our globals */
+tape_info_t *first_tape_info = NULL;
+tape_discipline_t *first_discipline = NULL;
+tape_frontend_t *first_frontend = NULL;
+#ifdef TAPE_DEBUG
+debug_info_t *tape_debug_area = NULL;
+#endif
+
+char* state_verbose[TS_SIZE]={
+    "TS_UNUSED",  "TS_IDLE", "TS_DONE", "TS_FAILED",
+    "TS_BLOCK_INIT",
+    "TS_BSB_INIT",
+    "TS_BSF_INIT",
+    "TS_DSE_INIT",
+    "TS_EGA_INIT",
+    "TS_FSB_INIT",
+    "TS_FSF_INIT",
+    "TS_LDI_INIT",
+    "TS_LBL_INIT",
+    "TS_MSE_INIT",
+    "TS_NOP_INIT",
+    "TS_RBA_INIT",
+    "TS_RBI_INIT",
+    "TS_RBU_INIT",
+    "TS_RBL_INIT",
+    "TS_RDC_INIT",
+    "TS_RFO_INIT",
+    "TS_RSD_INIT",
+    "TS_REW_INIT",
+    "TS_REW_RELEASE_INIT",
+    "TS_RUN_INIT",
+    "TS_SEN_INIT",
+    "TS_SID_INIT",
+    "TS_SNP_INIT",
+    "TS_SPG_INIT",
+    "TS_SWI_INIT",
+    "TS_SMR_INIT",
+    "TS_SYN_INIT",
+    "TS_TIO_INIT",
+    "TS_UNA_INIT",
+    "TS_WRI_INIT",
+    "TS_WTM_INIT",
+    "TS_NOT_OPER"};
+
+char* event_verbose[TE_SIZE]= {
+    "TE_START", "TE_DONE", "TE_FAILED", "TE_ERROR", "TE_OTHER"};
+
+#ifdef CONFIG_PROC_FS		/* don't waste space if unused */
+/*
+ * The proc filesystem: function to read and entry
+ */
+int
+tape_read_procmem (char *buf, char **start, off_t offset,
+		   int len, int unused)
+{
+	tape_info_t *tape = first_tape_info;
+	len = sprintf (buf, "minor\tstate\n");
+	do {
+		if (len >= PAGE_SIZE - 80)
+			len += sprintf (buf + len, "terminated...\n");
+		len += sprintf (buf + len,
+				"%d\t%s\n", tape->rew_minor,
+				((tapestate_get (tape) >= 0) && (tapestate_get (tape) < TS_SIZE)) ?
+				state_verbose[tapestate_get (tape)] :
+				"UNKNOWN STATE");
+	} while ((tape = (tape_info_t *) (tape->next)) != NULL);
+	return len;
+}
+
+#endif				/* CONFIG_PROC_FS */
+
+/* SECTION: Managing wrappers for ccwcache */
+
+#define TAPE_EMERGENCY_REQUESTS 16
+
+static ccw_req_t *tape_emergency_req[TAPE_EMERGENCY_REQUESTS] =
+{NULL,};
+static spinlock_t tape_emergency_req_lock = SPIN_LOCK_UNLOCKED;
+
+static void
+tape_init_emergency_req (void)
+{
+	int i;
+	for (i = 0; i < TAPE_EMERGENCY_REQUESTS; i++) {
+		tape_emergency_req[i] = (ccw_req_t *) get_free_page (GFP_KERNEL);
+	}
+}
+
+#ifdef MODULE // We only cleanup the emergency requests on module unload.
+static void
+tape_cleanup_emergency_req (void)
+{
+	int i;
+	for (i = 0; i < TAPE_EMERGENCY_REQUESTS; i++) {
+		if (tape_emergency_req[i])
+			free_page ((long) (tape_emergency_req[i]));
+		else
+			printk (KERN_WARNING PRINTK_HEADER "losing one page for 'in-use' emergency request\n");
+	}
+}
+#endif
+
+ccw_req_t *
+tape_alloc_request (char *magic, int cplength, int datasize)
+{
+	ccw_req_t *rv = NULL;
+	int i;
+	if ((rv = ccw_alloc_request (magic, cplength, datasize)) != NULL) {
+		return rv;
+	}
+	if (cplength * sizeof (ccw1_t) + datasize + sizeof (ccw_req_t) > PAGE_SIZE) {
+		return NULL;
+	}
+	spin_lock (&tape_emergency_req_lock);
+	for (i = 0; i < TAPE_EMERGENCY_REQUESTS; i++) {
+		if (tape_emergency_req[i] != NULL) {
+			rv = tape_emergency_req[i];
+			tape_emergency_req[i] = NULL;
+		}
+	}
+	spin_unlock (&tape_emergency_req_lock);
+	if (rv) {
+		memset (rv, 0, PAGE_SIZE);
+		rv->cache = (kmem_cache_t *) (tape_emergency_req + i);
+		strncpy ((char *) (&rv->magic), magic, 4);
+		ASCEBC ((char *) (&rv->magic), 4);
+		rv->cplength = cplength;
+		rv->datasize = datasize;
+		rv->data = (void *) ((long) rv + PAGE_SIZE - datasize);
+		rv->cpaddr = (ccw1_t *) ((long) rv + sizeof (ccw_req_t));
+	}
+	return rv;
+}
+
+void
+tape_free_request (ccw_req_t * request)
+{
+	if (request->cache >= (kmem_cache_t *) tape_emergency_req &&
+	    request->cache <= (kmem_cache_t *) (tape_emergency_req + TAPE_EMERGENCY_REQUESTS)) {
+		*((ccw_req_t **) (request->cache)) = request;
+	} else {
+		clear_normalized_cda ((ccw1_t *) (request->cpaddr));	// avoid memory leak caused by modeset_byte
+
+		ccw_free_request (request);
+	}
+}
+
+/*
+ * Allocate a ccw request and reserve it for tape driver
+ */
+inline
+ ccw_req_t *
+tape_alloc_ccw_req (tape_info_t * tape, int cplength, int datasize)
+{
+	char tape_magic_id[] = "tape";
+	ccw_req_t *cqr = NULL;
+
+	if (!tape)
+		return NULL;
+	cqr = tape_alloc_request (tape_magic_id, cplength, datasize);
+
+	if (!cqr) {
+#ifdef TAPE_DEBUG
+		PRINT_WARN ("empty CQR generated\n");
+#endif
+	}
+	cqr->magic = TAPE_MAGIC;	/* sets an identifier for tape driver   */
+	cqr->device = tape;	/* save pointer to tape info    */
+	return cqr;
+}
+
+/*
+ * Find the tape_info_t structure associated with irq
+ */
+static inline tape_info_t *
+tapedev_find_info (int irq)
+{
+	tape_info_t *tape;
+
+	tape = first_tape_info;
+	if (tape != NULL)
+		do {
+			if (tape->devinfo.irq == irq)
+				break;
+		} while ((tape = (tape_info_t *) tape->next) != NULL);
+	return tape;
+}
+
+#define QUEUE_THRESHOLD 5
+
+/*
+ * Tape interrupt routine, called from Ingo's I/O layer
+ */
+void
+tape_irq (int irq, void *int_parm, struct pt_regs *regs)
+{
+	tape_info_t *tape = tapedev_find_info (irq);
+
+	/* analyse devstat and fire event */
+	if (tape->devstat.dstat & DEV_STAT_UNIT_CHECK) {
+		tapestate_event (tape, TE_ERROR);
+	} else if (tape->devstat.dstat & (DEV_STAT_DEV_END)) {
+		tapestate_event (tape, TE_DONE);
+	} else
+		tapestate_event (tape, TE_OTHER);
+}
+
+int 
+tape_oper_handler ( int irq, struct _devreg *dreg) {
+    tape_info_t* tape=first_tape_info;
+    tape_info_t* newtape;
+    int rc,tape_num,retries=0;
+    s390_dev_info_t dinfo;
+    tape_discipline_t* disc;
+    long lockflags;
+    while ((tape!=NULL) && (tape->devinfo.irq!=irq)) 
+        tape=tape->next;
+    if (tape!=NULL) {
+        // irq is (still) used by tape. tell ingo to try again later
+        PRINT_WARN ("Oper handler for irq %d called while irq still (internaly?) used.\n",irq);
+        return -EAGAIN;
+    }
+    // irq is not used by tape
+    rc = get_dev_info_by_irq (irq, &dinfo);
+    if (rc == -ENODEV) {
+        retries++;
+        if (retries > 5)
+            return -ENODEV;
+    }
+    disc = first_discipline;
+    while ((disc != NULL) && (disc->cu_type != dinfo.sid_data.cu_type))
+        disc = (tape_discipline_t *) (disc->next);
+    
+    if ((disc == NULL) || (rc == -ENODEV))
+        return -ENODEV;
+    
+    /* Allocate tape structure  */
+    tape = kmalloc (sizeof (tape_info_t), GFP_ATOMIC);
+    if (tape == NULL) {
+        PRINT_INFO (KERN_ERR "tape: can't allocate memory for "
+                    "tape info structure\n");
+        return -ENOBUFS;
+    }
+    tape->discipline = disc;
+    disc->tape = tape;
+    tape_num=0;
+    newtape=first_tape_info;
+    while (newtape!=NULL) {
+        if (newtape->rew_minor==tape_num) {
+            // tape num in use. try next one
+            tape_num+=2;
+            newtape=first_tape_info;
+        } else {
+            // tape num not used by newtape. look at next tape info
+            newtape=newtape->next;
+        }
+    }
+    rc = tape_setup (tape, irq, tape_num);
+    if (rc) {
+        kfree (tape);
+        return -ENOBUFS;
+    }
+    s390irq_spin_lock_irqsave (irq,lockflags);
+    if (first_tape_info == NULL) {
+        first_tape_info = tape;
+    } else {
+        newtape = first_tape_info;
+        while (newtape->next != NULL)
+            newtape = newtape->next;
+        newtape->next = tape;
+    }
+    s390irq_spin_unlock_irqrestore (irq, lockflags);
+    return 0;
+}
+
+
+static void
+tape_noper_handler ( int irq, int status ) {
+    tape_info_t *ti=first_tape_info;
+    tape_info_t *lastti;
+    long lockflags;
+    s390irq_spin_lock_irqsave(irq,lockflags);
+    while (ti!=NULL) {
+        if (ti->devinfo.irq==irq) {
+            tapestate_set(ti,TS_NOT_OPER);
+            if (tapestate_get(ti)!=TS_UNUSED) {
+                // device is in use!
+                PRINT_WARN ("Tape #%d was detached while it was busy. Expect errors!",ti->blk_minor);
+                ti->rc=-ENXIO; 
+                wake_up_interruptible(&ti->wq);
+            } else {
+                // device is unused!
+                if (ti==first_tape_info) {
+                    first_tape_info=ti->next;
+                } else {
+                    lastti=first_tape_info;
+                    while (lastti->next!=ti) lastti=lastti->next;
+                    lastti->next=ti->next;
+                }
+                kfree(ti);
+            }
+            s390irq_spin_unlock_irqrestore(irq,lockflags);
+            return;
+        }
+        ti=ti->next;
+    }
+    s390irq_spin_unlock_irqrestore(irq,lockflags);
+    PRINT_WARN ("Tape not found for irq %d. Device is detached.",irq);
+}
+
+
+void
+tape_dump_sense (devstat_t * stat)
+{
+	int sl, sct;
+	PRINT_WARN ("------------I/O resulted in unit check:-----------\n");
+	for (sl = 0; sl < 4; sl++) {
+		PRINT_WARN ("Sense:");
+		for (sct = 0; sct < 8; sct++) {
+			PRINT_WARN (" %2d:0x%02X", 8 * sl + sct,
+				    stat->ii.sense.data[8 * sl + sct]);
+		}
+		PRINT_WARN ("\n");
+	}
+	PRINT_INFO ("Sense data: %02X%02X%02X%02X %02X%02X%02X%02X "
+		    " %02X%02X%02X%02X %02X%02X%02X%02X \n",
+		    stat->ii.sense.data[0], stat->ii.sense.data[1],
+		    stat->ii.sense.data[2], stat->ii.sense.data[3],
+		    stat->ii.sense.data[4], stat->ii.sense.data[5],
+		    stat->ii.sense.data[6], stat->ii.sense.data[7],
+		    stat->ii.sense.data[8], stat->ii.sense.data[9],
+		    stat->ii.sense.data[10], stat->ii.sense.data[11],
+		    stat->ii.sense.data[12], stat->ii.sense.data[13],
+		    stat->ii.sense.data[14], stat->ii.sense.data[15]);
+	PRINT_INFO ("Sense data: %02X%02X%02X%02X %02X%02X%02X%02X "
+		    " %02X%02X%02X%02X %02X%02X%02X%02X \n",
+		    stat->ii.sense.data[16], stat->ii.sense.data[17],
+		    stat->ii.sense.data[18], stat->ii.sense.data[19],
+		    stat->ii.sense.data[20], stat->ii.sense.data[21],
+		    stat->ii.sense.data[22], stat->ii.sense.data[23],
+		    stat->ii.sense.data[24], stat->ii.sense.data[25],
+		    stat->ii.sense.data[26], stat->ii.sense.data[27],
+		    stat->ii.sense.data[28], stat->ii.sense.data[29],
+		    stat->ii.sense.data[30], stat->ii.sense.data[31]);
+#ifdef TAPE_DEBUG
+        debug_text_event (tape_debug_area,3,"SENSE:");
+        for (sl=0;sl<31;sl++) {
+            debug_int_event (tape_debug_area,3,stat->ii.sense.data[sl]);
+        }
+        debug_int_exception (tape_debug_area,3,stat->ii.sense.data[31]);
+#endif
+}
+
+/*
+ * Setup tape_info_t structure of a tape device
+ */
+int
+tape_setup (tape_info_t * ti, int irq, int minor)
+{
+	long lockflags;
+	int rc = 0;
+
+	rc = get_dev_info_by_irq (irq, &(ti->devinfo));
+	if (rc == -ENODEV) {	/* end of device list */
+		return rc;
+	}
+	ti->rew_minor = minor;
+	ti->nor_minor = minor + 1;
+	ti->blk_minor = minor;
+	/* Register IRQ */
+#ifdef CONFIG_S390_TAPE_DYNAMIC
+        rc = s390_request_irq_special (irq, tape_irq, tape_noper_handler,0, "tape", &(ti->devstat));
+#else
+	rc = s390_request_irq (irq, tape_irq, 0, "tape", &(ti->devstat));
+#endif
+	s390irq_spin_lock_irqsave (irq, lockflags);
+	ti->next = NULL;
+	if (rc) {
+		PRINT_WARN ("Cannot register irq %d, rc=%d\n", irq, rc);
+	} else
+		PRINT_WARN ("Register irq %d for using with discipline %x\n", irq, ti->discipline->cu_type);
+	init_waitqueue_head (&ti->wq);
+	ti->kernbuf = ti->userbuf = ti->discdata = NULL;
+	tapestate_set (ti, TS_UNUSED);
+        ti->discdata=NULL;
+	ti->discipline->setup_assist (ti);
+        ti->wanna_wakeup=0;
+	s390irq_spin_unlock_irqrestore (irq, lockflags);
+	return rc;
+}
+
+/*
+ *      tape_init will register the driver for each tape.
+ */
+int
+tape_init (void)
+{
+	long lockflags;
+	s390_dev_info_t dinfo;
+	tape_discipline_t *disc;
+	tape_info_t *ti = NULL, *tempti = NULL;
+        char *opt_char,*opt_block,*opt_3490,*opt_3480;
+	int irq = 0, rc, retries = 0, tape_num = 0;
+        static int initialized=0;
+
+        if (initialized) // Only init the devices once
+            return 0;
+        initialized=1;
+
+#ifdef TAPE_DEBUG
+        tape_debug_area = debug_register ( "tape", 3, 2, 10);
+        debug_register_view(tape_debug_area,&debug_hex_ascii_view);
+        debug_text_event (tape_debug_area,3,"begin init");
+#endif /* TAPE_DEBUG */
+
+        /* print banner */        
+        PRINT_WARN ("IBM S/390 Tape Device Driver (ALPHA).\n");
+        PRINT_WARN ("(C) IBM Deutschland Entwicklung GmbH, 2000\n");
+        opt_char=opt_block=opt_3480=opt_3490="not present";
+#ifdef CONFIG_S390_TAPE_CHAR
+        opt_char="built in";
+#endif
+#ifdef CONFIG_S390_TAPE_BLOCK
+        opt_block="built in";
+#endif
+#ifdef CONFIG_S390_TAPE_3480
+        opt_3480="built in";
+#endif
+#ifdef CONFIG_S390_TAPE_3490
+        opt_3490="built in";
+#endif
+        /* print feature info */
+        PRINT_WARN ("character device frontend   : %s\n",opt_char);
+        PRINT_WARN ("block device frontend       : %s\n",opt_block);
+        PRINT_WARN ("support for 3480 compatible : %s\n",opt_3480);
+        PRINT_WARN ("support for 3490 compatible : %s\n",opt_3490);
+        
+
+#ifdef CONFIG_S390_TAPE_3490
+	first_discipline = tape3490_init ();
+	first_discipline->next = NULL;
+#endif
+
+#ifdef CONFIG_S390_TAPE_3480
+        if (first_discipline == NULL) {
+            first_discipline = tape3480_init ();
+            first_discipline->next = NULL;
+        } else {
+            first_discipline->next = tape3480_init ();
+            ((tape_discipline_t*) (first_discipline->next))->next=NULL;
+        }
+#endif
+
+#ifdef TAPE_DEBUG
+        debug_text_event (tape_debug_area,3,"dev detect");
+#endif /* TAPE_DEBUG */
+	/* Allocate the tape structures */
+	for (irq = 0; irq < NR_IRQS; irq++) {
+		rc = get_dev_info_by_irq (irq, &dinfo);
+
+		if (rc == -ENODEV) {
+			retries++;
+			if (retries > 5)
+				irq = NR_IRQS;
+		}
+		disc = first_discipline;
+		while ((disc != NULL) && (disc->cu_type != dinfo.sid_data.cu_type))
+			disc = (tape_discipline_t *) (disc->next);
+
+		if ((disc == NULL) || (rc == -ENODEV))
+			continue;
+#ifdef TAPE_DEBUG
+                debug_text_event (tape_debug_area,3,"det irq:  ");
+                debug_int_event (tape_debug_area,3,irq);
+                debug_text_event (tape_debug_area,3,"cu:       ");
+                debug_int_event (tape_debug_area,3,disc->cu_type);
+#endif /* TAPE_DEBUG */
+		/* Allocate tape structure  */
+		ti = kmalloc (sizeof (tape_info_t), GFP_ATOMIC);
+		if (ti == NULL) {
+#ifdef TAPE_DEBUG
+                    debug_text_exception (tape_debug_area,3,"ti:no mem ");
+#endif /* TAPE_DEBUG */
+                    PRINT_INFO (KERN_ERR "tape: can't allocate memory for "
+				    "tape info structure\n");
+                    continue;
+		}
+		memset(ti,0,sizeof(tape_info_t));
+		ti->discipline = disc;
+		disc->tape = ti;
+		rc = tape_setup (ti, irq, tape_num);
+		if (rc) {
+#ifdef TAPE_DEBUG
+                        debug_text_event (tape_debug_area,3,"tsetup err");
+                        debug_int_exception (tape_debug_area,3,rc);
+#endif /* TAPE_DEBUG */
+			kfree (ti);
+		} else {
+			s390irq_spin_lock_irqsave (irq, lockflags);
+			if (first_tape_info == NULL) {
+				first_tape_info = ti;
+			} else {
+				tempti = first_tape_info;
+				while (tempti->next != NULL)
+					tempti = tempti->next;
+				tempti->next = ti;
+			}
+			tape_num += 2;
+			s390irq_spin_unlock_irqrestore (irq, lockflags);
+		}
+	}
+
+	/* Allocate local buffer for the ccwcache       */
+	tape_init_emergency_req ();
+
+#if 0				// need to register s.th. for proc? FIXME!
+	if (proc_register_dynamic (&proc_root, &proc_root_tape))
+		PRINT_INFO (KERN_ERR "tape: registering "
+			    "/proc/tape failed\n");
+#endif
+
+	return 0;
+}
+
+#ifdef MODULE
+int
+init_module (void)
+{
+#ifdef CONFIG_S390_TAPE_CHAR
+	tapechar_init ();
+#endif
+#ifdef CONFIG_S390_TAPE_BLOCK
+	tapeblock_init ();
+#endif
+	return 0;
+}
+
+void
+cleanup_module (void)
+{
+        tape_info_t *tape ,*temp;
+        tape_frontend_t* frontend, *tempfe;
+        tape_discipline_t* disc ,*tempdi;
+#ifdef TAPE_DEBUG
+        debug_text_event (tape_debug_area,6,"cleaup mod");
+#endif /* TAPE_DEBUG */
+
+#if 0				/* FIXME: Do we need to register s.th? */
+	proc_unregister (&proc_root, proc_root_tape.low_ino);
+#endif
+
+	tape = first_tape_info;
+	while (tape != NULL) {
+		temp = tape;
+		tape = tape->next;
+                //cleanup a device 
+#ifdef TAPE_DEBUG
+                debug_text_event (tape_debug_area,6,"free irq:");
+                debug_int_event (tape_debug_area,6,temp->devinfo.irq);
+#endif /* TAPE_DEBUG */
+		free_irq (temp->devinfo.irq, &(temp->devstat));
+                if (temp->discdata) kfree (temp->discdata);
+                if (temp->kernbuf) kfree (temp->kernbuf);
+#if (LINUX_VERSION_CODE > KERNEL_VERSION(2,3,98)) // COMPAT:FIXME
+#else
+                kfree (temp->wq);
+#endif
+                if (temp->cqr) tape_free_request(temp->cqr);
+		kfree (temp);
+	}
+#ifdef CONFIG_S390_TAPE_CHAR
+	tapechar_uninit();
+#endif
+#ifdef CONFIG_S390_TAPE_BLOCK
+        tapeblock_uninit();
+#endif
+        frontend=first_frontend;
+	while (frontend != NULL) {
+		tempfe = frontend;
+		frontend = frontend->next;
+		kfree (tempfe);
+	}
+        disc=first_discipline;
+	while (disc != NULL) {
+		tempdi = disc;
+		disc = disc->next;
+		kfree (tempdi);
+	}
+	/* Deallocate the local buffer for the ccwcache         */
+	tape_cleanup_emergency_req ();
+#ifdef TAPE_DEBUG
+        debug_unregister (tape_debug_area);
+#endif /* TAPE_DEBUG */
+}
+#endif				/* MODULE */
+
+inline void
+tapestate_set (tape_info_t * tape, int newstate)
+{
+    if (tape->tape_state == TS_NOT_OPER) {
+#ifdef TAPE_DEBUG
+        debug_text_event (tape_debug_area,3,"ts_set err");
+        debug_text_exception (tape_debug_area,3,"dev n.oper");
+#endif /* TAPE_DEBUG */
+    } else {
+#ifdef TAPE_DEBUG
+        debug_text_event (tape_debug_area,4,"ts. dev:  ");
+        debug_int_event (tape_debug_area,4,tape->blk_minor);
+        debug_text_event (tape_debug_area,4,"old ts:   ");
+        debug_text_event (tape_debug_area,4,(((tapestate_get (tape) < TS_SIZE) &&
+                                             (tapestate_get (tape) >=0 )) ?
+                                            state_verbose[tapestate_get (tape)] :
+                                            "UNKNOWN TS"));
+        debug_text_event (tape_debug_area,4,"new ts:   ");
+        debug_text_event (tape_debug_area,4,(((newstate < TS_SIZE) &&
+                                              (newstate >= 0)) ?
+                                             state_verbose[newstate] :
+                                             "UNKNOWN TS"));
+#endif /* TAPE_DEBUG */
+	tape->tape_state = newstate;
+    }
+}
+
+inline int
+tapestate_get (tape_info_t * tape)
+{
+	return (tape->tape_state);
+}
+
+void
+tapestate_event (tape_info_t * tape, int event)
+{
+#ifdef TAPE_DEBUG
+        debug_text_event (tape_debug_area,6,"te! dev:  ");
+        debug_int_event (tape_debug_area,6,tape->blk_minor);
+        debug_text_event (tape_debug_area,6,"event:");
+        debug_text_event (tape_debug_area,6,((event >=0) &&
+                                            (event < TE_SIZE)) ?
+                         event_verbose[event] : "TE UNKNOWN");
+        debug_text_event (tape_debug_area,6,"state:");
+        debug_text_event (tape_debug_area,6,((tapestate_get(tape) >= 0) &&
+                                            (tapestate_get(tape) < TS_SIZE)) ?
+                         state_verbose[tapestate_get (tape)] :
+                         "TS UNKNOWN");
+#endif /* TAPE_DEBUG */    
+	if ((event >= 0) &&
+	    (event < TE_SIZE) &&
+	    (tapestate_get (tape) >= 0) &&
+	    (tapestate_get (tape) < TS_SIZE) &&
+	    ((*(tape->discipline->event_table))[tapestate_get (tape)][event] != NULL))
+		((*(tape->discipline->event_table))[tapestate_get (tape)][event]) (tape);
+	else {
+#ifdef TAPE_DEBUG
+                debug_text_exception (tape_debug_area,3,"TE UNEXPEC");
+#endif /* TAPE_DEBUG */
+		tape->discipline->default_handler (tape);
+	}
+}
+
+/*
+ * Overrides for Emacs so that we follow Linus's tabbing style.
+ * Emacs will notice this stuff at the end of the file and automatically
+ * adjust the settings for this buffer only.  This must remain at the end
+ * of the file.
+ * ---------------------------------------------------------------------------
+ * Local variables:
+ * c-indent-level: 4
+ * c-brace-imaginary-offset: 0
+ * c-brace-offset: -4
+ * c-argdecl-indent: 4
+ * c-label-offset: -4
+ * c-continued-statement-offset: 4
+ * c-continued-brace-offset: 0
+ * indent-tabs-mode: nil
+ * tab-width: 8
+ * End:
+ */

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