Index: nozomi-060209/nozomi.c
===================================================================
--- nozomi-060209.orig/nozomi.c	2007-06-04 13:22:47.338658712 +0200
+++ nozomi-060209/nozomi.c	2007-06-04 13:22:47.527629984 +0200
@@ -7,6 +7,9 @@
  *
  * Maintained by: Paul Hardwick, p.hardwick@option.com
  *
+ * Patches:
+ *          Locking code changes for Vodafone, Andrew Bird & Phil Sanderson
+ *
  * Source has been ported from an implementation made by Filip Aben, f.aben@option.com
  *
  * --------------------------------------------------------------------------
@@ -61,6 +64,7 @@
 #include <linux/interrupt.h>
 #include <linux/kmod.h>
 #include <linux/proc_fs.h>
+#include <linux/init.h>
 #include <asm/uaccess.h>
 
 
@@ -133,23 +137,23 @@
 /* TODO: rewrite to optimize macros... */
 #define SET_FCR(value__) \
   do {  \
-	writew((value__), (void*) (dc->REG_FCR )); \
+	writew((value__), (dc->REG_FCR )); \
 } while(0)
 
 #define SET_IER(value__, mask__) \
   do {  \
 	dc->ier_last_written = (dc->ier_last_written & ~mask__) | (value__ & mask__ );\
-    writew( dc->ier_last_written, (void*) (dc->REG_IER));\
+    writew( dc->ier_last_written, (dc->REG_IER));\
 } while(0)
 
 #define GET_IER(read_val__) \
   do {  \
-	(read_val__) = readw((void*) (dc->REG_IER));\
+	(read_val__) = readw((dc->REG_IER));\
 } while(0)
 
 #define GET_IIR(read_val__) \
   do {  \
-	(read_val__) = readw((void*) (dc->REG_IIR));\
+	(read_val__) = readw( (dc->REG_IIR));\
 } while(0)
 
 #define GET_MEM(value__, addr__, length__) \
@@ -265,7 +269,7 @@
 /* There are two types of nozomi cards, one with 2048 memory and with 8192 memory */
 typedef enum {
 	F32_2 = 2048, /* Has 512 bytes downlink and uplink * 2             -> 2048 */
-	F32_8 = 9192, /* Has 3072 bytes downlink and 1024 bytes uplink * 2 -> 8192 */
+	F32_8 = 8192, /* Has 3072 bytes downlink and 1024 bytes uplink * 2 -> 8192 */
 } card_type_t;
 
 /* Two different toggle channels exist */
@@ -438,12 +442,12 @@
 	u32				     base_addr;
 	u8				     closing;
 
-	/* Register addresses */
-	u32				     REG_IIR;
-	u32				     REG_FCR;
-	u32				     REG_IER;
+   /* Pointers to registers ( register is tagged volatile, not pointer ) */
+	volatile u16 * REG_IIR;
+	volatile u16 * REG_FCR;
+	volatile u16 * REG_IER;
 
-	volatile u16		 ier_last_written;
+	u16		 ier_last_written;
 	card_type_t 	     card_type;
 	config_table_t 	     config_table; /* Configuration table */
 	struct pci_dev 	    *pdev;
@@ -490,7 +494,7 @@
 
 /* Used to store interrupt variables */
 typedef struct {
-	volatile u16 read_iir; /* Holds current interrupt tokens */
+	u16 read_iir; /* Holds current interrupt tokens */
 } irq_t;
 
 MODULE_DEVICE_TABLE(pci, nozomi_pci_tbl);
@@ -1345,9 +1349,9 @@
 	u32 offset = dc->base_addr + dc->card_type/2;
 	int i;
 
-	dc->REG_FCR = offset + R_FCR;
-	dc->REG_IIR = offset + R_IIR;
-	dc->REG_IER = offset + R_IER;
+	dc->REG_FCR = (u16 *) (offset + R_FCR);
+	dc->REG_IIR = (u16 *) (offset + R_IIR);
+	dc->REG_IER = (u16 *) (offset + R_IER);
 	dc->ier_last_written = 0;
 	dc->closing = 0;
 
@@ -1366,13 +1370,16 @@
 static void tty_flip_queue_function(void *tmp_dc) {
 	dc_t *dc = (dc_t*) tmp_dc;
 	int i;
+   u32 flags;
 
 	/* Enable interrupt for that port */
 	for(i=0;i<MAX_PORT;i++) {
 		if (dc->port[i].tty_dont_flip) {
 			D6("Enable for port: %d", i);
 			dc->port[i].tty_dont_flip = 0;
+         spin_lock_irqsave(&dc->spin_mutex, flags);
 			enable_transmit_dl(dc->port[i].tty_index, dc);
+         spin_unlock_irqrestore(&dc->spin_mutex, flags);
 		}
 	}
 }
@@ -1555,7 +1562,11 @@
 
 static void tty_do_close(dc_t *dc, port_t *port) {
 
-	down(&port->tty_sem);
+	u32      flags;
+
+	if(down_interruptible(&port->tty_sem)){
+      return;
+   }
 
 	if ( !port->tty_open_count ) {
 		goto exit;
@@ -1569,7 +1580,9 @@
 
 	if ( port->tty_open_count == 0) {
         D1("close: %d", port->token_dl );
+      spin_lock_irqsave(&dc->spin_mutex, flags);
 		SET_IER( 0,  port->token_dl );
+      spin_unlock_irqrestore(&dc->spin_mutex, flags);
 	}
 
 exit:
@@ -1679,8 +1692,11 @@
 	s32 index = get_index(tty);
 	port_t *port = get_port_by_tty(tty);
 	dc_t *dc = get_dc_by_tty(tty);
+	u32      flags;
 
-	down(&port->tty_sem);
+   if(down_interruptible(&port->tty_sem)){
+      return -ERESTARTSYS;
+   }
 
 	tty->low_latency  = 1;
 	tty->driver_data  = port;
@@ -1698,7 +1714,9 @@
 	if ( port->tty_open_count == 1) {
 		port->rx_data = port->tx_data = 0;
         D1("open: %d", port->token_dl );
+      spin_lock_irqsave(&dc->spin_mutex, flags);
 		SET_IER( port->token_dl,  port->token_dl );
+      spin_unlock_irqrestore(&dc->spin_mutex, flags);
 	}
 
 	up(&port->tty_sem);
@@ -1722,6 +1740,7 @@
 	int                  rval = -EINVAL;
 	dc_t                *dc = get_dc_by_tty(tty);
 	port_t              *port = (port_t *) tty->driver_data;
+	u32      flags;
 
 	/* D1( "WRITEx: %d, index = %d", count, index); */
 
@@ -1729,7 +1748,10 @@
 		return -ENODEV;
 	}
 
-	down(&port->tty_sem);
+   if(down_trylock(&port->tty_sem) ) { // must test lock as tty layer wraps calls to this function with BKL
+      ERR("Would have deadlocked - return ERESTARTSYS");
+      return -ERESTARTSYS;
+   }
 
 	if (! port->tty_open_count) {
         D1( " ");
@@ -1752,6 +1774,7 @@
 		goto exit;
 	}
 
+   spin_lock_irqsave(&dc->spin_mutex, flags);
 	// CTS is only valid on the modem channel
 	if ( port == &(dc->port[PORT_MDM]) ) {
 		if ( port->ctrl_dl.CTS ) {
@@ -1763,6 +1786,7 @@
 	} else {
 		enable_transmit_ul(port->tty_index, dc );
 	}
+   spin_unlock_irqrestore(&dc->spin_mutex, flags);
 
 exit:
 	up(&port->tty_sem);
@@ -1782,7 +1806,9 @@
         return 0;
 	}
 
-	down(&port->tty_sem);
+   if(down_interruptible(&port->tty_sem)){
+      return 0;
+   }
 
 	if (! port->tty_open_count) {
 		goto exit;
@@ -1969,6 +1995,8 @@
 
 static int ntty_ioctl(struct tty_struct *tty, struct file *file, unsigned int cmd, unsigned long arg) {
 	port_t *port = (port_t *) tty->driver_data;
+	dc_t    *dc = get_dc_by_tty(tty);
+	u32      flags;
 	int mask;
 	int rval = -ENOIOCTLCMD;
 
@@ -1991,7 +2019,9 @@
 		rval =  ntty_ioctl_tiocgicount(tty, file, cmd, arg);
 		break;
 	case TIOCMGET:
+      spin_lock_irqsave(&dc->spin_mutex, flags);
 		rval = ntty_tiocmget(tty, file);
+      spin_unlock_irqrestore(&dc->spin_mutex, flags);
 		break;
 	case TIOCMSET:
 		rval = ntty_tiocmset(tty, file, arg);
@@ -2000,20 +2030,24 @@
 		if (get_user(mask, (unsigned long *) arg))
 			return -EFAULT;
 
+      spin_lock_irqsave(&dc->spin_mutex, flags);
 		if (mask & TIOCM_RTS)
 			set_rts(port->tty_index, 0);
 		if (mask & TIOCM_DTR)
 			set_dtr(port->tty_index, 0);
+      spin_unlock_irqrestore(&dc->spin_mutex, flags);
 		rval = 0;
 		break;
    	case TIOCMBIS:
 		if (get_user(mask, (unsigned long *) arg))
 			return -EFAULT;
 
+      spin_lock_irqsave(&dc->spin_mutex, flags);
 		if (mask & TIOCM_RTS)
 			set_rts(port->tty_index, 1);
 		if (mask & TIOCM_DTR)
 			set_dtr(port->tty_index, 1);
+      spin_unlock_irqrestore(&dc->spin_mutex, flags);
 		rval = 0;
 		break;
 	case TCFLSH: