mirror of
				git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
				synced 2025-10-31 08:44:41 +00:00 
			
		
		
		
	atari_scsi: Fix atari_scsi deadlocks on Falcon
Don't disable irqs when waiting for the ST DMA "lock"; its release may require an interrupt. Introduce stdma_try_lock() for use in soft irq context. atari_scsi now tells the SCSI mid-layer to defer queueing a command if the ST DMA lock is not available, as per Michael's patch: http://marc.info/?l=linux-m68k&m=139095335824863&w=2 The falcon_got_lock variable is race prone: we can't disable IRQs while waiting to acquire the lock, so after acquiring it there must be some interval during which falcon_got_lock remains false. Introduce stdma_is_locked_by() to replace falcon_got_lock. The falcon_got_lock tests in the EH handlers are incorrect these days. It can happen that an EH handler is called after a command completes normally. Remove these checks along with falcon_got_lock. Also remove the complicated and racy fairness wait queues. If fairness is an issue (when SCSI competes with IDE for the ST DMA interrupt), the solution is likely to be a lower value for host->can_queue. Signed-off-by: Finn Thain <fthain@telegraphics.com.au> Reviewed-by: Hannes Reinecke <hare@suse.de> Tested-by: Michael Schmitz <schmitzmic@gmail.com> Acked-by: Geert Uytterhoeven <geert@linux-m68k.org> Signed-off-by: Christoph Hellwig <hch@lst.de>
This commit is contained in:
		
							parent
							
								
									cbad48deb3
								
							
						
					
					
						commit
						16b29e75a7
					
				
					 4 changed files with 66 additions and 107 deletions
				
			
		|  | @ -59,6 +59,31 @@ static irqreturn_t stdma_int (int irq, void *dummy); | |||
| /************************* End of Prototypes **************************/ | ||||
| 
 | ||||
| 
 | ||||
| /**
 | ||||
|  * stdma_try_lock - attempt to acquire ST DMA interrupt "lock" | ||||
|  * @handler: interrupt handler to use after acquisition | ||||
|  * | ||||
|  * Returns !0 if lock was acquired; otherwise 0. | ||||
|  */ | ||||
| 
 | ||||
| int stdma_try_lock(irq_handler_t handler, void *data) | ||||
| { | ||||
| 	unsigned long flags; | ||||
| 
 | ||||
| 	local_irq_save(flags); | ||||
| 	if (stdma_locked) { | ||||
| 		local_irq_restore(flags); | ||||
| 		return 0; | ||||
| 	} | ||||
| 
 | ||||
| 	stdma_locked   = 1; | ||||
| 	stdma_isr      = handler; | ||||
| 	stdma_isr_data = data; | ||||
| 	local_irq_restore(flags); | ||||
| 	return 1; | ||||
| } | ||||
| EXPORT_SYMBOL(stdma_try_lock); | ||||
| 
 | ||||
| 
 | ||||
| /*
 | ||||
|  * Function: void stdma_lock( isrfunc isr, void *data ) | ||||
|  | @ -78,19 +103,10 @@ static irqreturn_t stdma_int (int irq, void *dummy); | |||
| 
 | ||||
| void stdma_lock(irq_handler_t handler, void *data) | ||||
| { | ||||
| 	unsigned long flags; | ||||
| 
 | ||||
| 	local_irq_save(flags);		/* protect lock */ | ||||
| 
 | ||||
| 	/* Since the DMA is used for file system purposes, we
 | ||||
| 	 have to sleep uninterruptible (there may be locked | ||||
| 	 buffers) */ | ||||
| 	wait_event(stdma_wait, !stdma_locked); | ||||
| 
 | ||||
| 	stdma_locked   = 1; | ||||
| 	stdma_isr      = handler; | ||||
| 	stdma_isr_data = data; | ||||
| 	local_irq_restore(flags); | ||||
| 	wait_event(stdma_wait, stdma_try_lock(handler, data)); | ||||
| } | ||||
| EXPORT_SYMBOL(stdma_lock); | ||||
| 
 | ||||
|  | @ -122,22 +138,25 @@ void stdma_release(void) | |||
| EXPORT_SYMBOL(stdma_release); | ||||
| 
 | ||||
| 
 | ||||
| /*
 | ||||
|  * Function: int stdma_others_waiting( void ) | ||||
|  * | ||||
|  * Purpose: Check if someone waits for the ST-DMA lock. | ||||
|  * | ||||
|  * Inputs: none | ||||
|  * | ||||
|  * Returns: 0 if no one is waiting, != 0 otherwise | ||||
| /**
 | ||||
|  * stdma_is_locked_by - allow lock holder to check whether it needs to release. | ||||
|  * @handler: interrupt handler previously used to acquire lock. | ||||
|  * | ||||
|  * Returns !0 if locked for the given handler; 0 otherwise. | ||||
|  */ | ||||
| 
 | ||||
| int stdma_others_waiting(void) | ||||
| int stdma_is_locked_by(irq_handler_t handler) | ||||
| { | ||||
| 	return waitqueue_active(&stdma_wait); | ||||
| 	unsigned long flags; | ||||
| 	int result; | ||||
| 
 | ||||
| 	local_irq_save(flags); | ||||
| 	result = stdma_locked && (stdma_isr == handler); | ||||
| 	local_irq_restore(flags); | ||||
| 
 | ||||
| 	return result; | ||||
| } | ||||
| EXPORT_SYMBOL(stdma_others_waiting); | ||||
| EXPORT_SYMBOL(stdma_is_locked_by); | ||||
| 
 | ||||
| 
 | ||||
| /*
 | ||||
|  |  | |||
|  | @ -8,11 +8,11 @@ | |||
| 
 | ||||
| /***************************** Prototypes *****************************/ | ||||
| 
 | ||||
| int stdma_try_lock(irq_handler_t, void *); | ||||
| void stdma_lock(irq_handler_t handler, void *data); | ||||
| void stdma_release( void ); | ||||
| int stdma_others_waiting( void ); | ||||
| int stdma_islocked( void ); | ||||
| void *stdma_locked_by( void ); | ||||
| int stdma_is_locked_by(irq_handler_t); | ||||
| void stdma_init( void ); | ||||
| 
 | ||||
| /************************* End of Prototypes **************************/ | ||||
|  |  | |||
|  | @ -879,10 +879,10 @@ static void NCR5380_exit(struct Scsi_Host *instance) | |||
|  * | ||||
|  */ | ||||
| 
 | ||||
| static int NCR5380_queue_command_lck(struct scsi_cmnd *cmd, | ||||
|                                      void (*done)(struct scsi_cmnd *)) | ||||
| static int NCR5380_queue_command(struct Scsi_Host *instance, | ||||
|                                  struct scsi_cmnd *cmd) | ||||
| { | ||||
| 	SETUP_HOSTDATA(cmd->device->host); | ||||
| 	struct NCR5380_hostdata *hostdata = shost_priv(instance); | ||||
| 	struct scsi_cmnd *tmp; | ||||
| 	unsigned long flags; | ||||
| 
 | ||||
|  | @ -893,7 +893,7 @@ static int NCR5380_queue_command_lck(struct scsi_cmnd *cmd, | |||
| 		printk(KERN_NOTICE "scsi%d: WRITE attempted with NO_WRITE debugging flag set\n", | ||||
| 		       H_NO(cmd)); | ||||
| 		cmd->result = (DID_ERROR << 16); | ||||
| 		done(cmd); | ||||
| 		cmd->scsi_done(cmd); | ||||
| 		return 0; | ||||
| 	} | ||||
| #endif /* (NDEBUG & NDEBUG_NO_WRITE) */ | ||||
|  | @ -904,8 +904,6 @@ static int NCR5380_queue_command_lck(struct scsi_cmnd *cmd, | |||
| 	 */ | ||||
| 
 | ||||
| 	SET_NEXT(cmd, NULL); | ||||
| 	cmd->scsi_done = done; | ||||
| 
 | ||||
| 	cmd->result = 0; | ||||
| 
 | ||||
| 	/*
 | ||||
|  | @ -915,7 +913,6 @@ static int NCR5380_queue_command_lck(struct scsi_cmnd *cmd, | |||
| 	 * sense data is only guaranteed to be valid while the condition exists. | ||||
| 	 */ | ||||
| 
 | ||||
| 	local_irq_save(flags); | ||||
| 	/* ++guenther: now that the issue queue is being set up, we can lock ST-DMA.
 | ||||
| 	 * Otherwise a running NCR5380_main may steal the lock. | ||||
| 	 * Lock before actually inserting due to fairness reasons explained in | ||||
|  | @ -928,11 +925,11 @@ static int NCR5380_queue_command_lck(struct scsi_cmnd *cmd, | |||
| 	 * because also a timer int can trigger an abort or reset, which would | ||||
| 	 * alter queues and touch the lock. | ||||
| 	 */ | ||||
| 	if (!IS_A_TT()) { | ||||
| 		/* perhaps stop command timer here */ | ||||
| 		falcon_get_lock(); | ||||
| 		/* perhaps restart command timer here */ | ||||
| 	} | ||||
| 	if (!falcon_get_lock()) | ||||
| 		return SCSI_MLQUEUE_HOST_BUSY; | ||||
| 
 | ||||
| 	local_irq_save(flags); | ||||
| 
 | ||||
| 	if (!(hostdata->issue_queue) || (cmd->cmnd[0] == REQUEST_SENSE)) { | ||||
| 		LIST(cmd, hostdata->issue_queue); | ||||
| 		SET_NEXT(cmd, hostdata->issue_queue); | ||||
|  | @ -956,15 +953,13 @@ static int NCR5380_queue_command_lck(struct scsi_cmnd *cmd, | |||
| 	 * If we're not in an interrupt, we can call NCR5380_main() | ||||
| 	 * unconditionally, because it cannot be already running. | ||||
| 	 */ | ||||
| 	if (in_interrupt() || ((flags >> 8) & 7) >= 6) | ||||
| 	if (in_interrupt() || irqs_disabled()) | ||||
| 		queue_main(); | ||||
| 	else | ||||
| 		NCR5380_main(NULL); | ||||
| 	return 0; | ||||
| } | ||||
| 
 | ||||
| static DEF_SCSI_QCMD(NCR5380_queue_command) | ||||
| 
 | ||||
| /*
 | ||||
|  * Function : NCR5380_main (void) | ||||
|  * | ||||
|  | @ -2554,10 +2549,6 @@ int NCR5380_abort(struct scsi_cmnd *cmd) | |||
| 
 | ||||
| 	local_irq_save(flags); | ||||
| 
 | ||||
| 	if (!IS_A_TT() && !falcon_got_lock) | ||||
| 		printk(KERN_ERR "scsi%d: !!BINGO!! Falcon has no lock in NCR5380_abort\n", | ||||
| 		       HOSTNO); | ||||
| 
 | ||||
| 	dprintk(NDEBUG_ABORT, "scsi%d: abort called basr 0x%02x, sr 0x%02x\n", HOSTNO, | ||||
| 		    NCR5380_read(BUS_AND_STATUS_REG), | ||||
| 		    NCR5380_read(STATUS_REG)); | ||||
|  | @ -2756,10 +2747,6 @@ static int NCR5380_bus_reset(struct scsi_cmnd *cmd) | |||
| 	struct scsi_cmnd *connected, *disconnected_queue; | ||||
| #endif | ||||
| 
 | ||||
| 	if (!IS_A_TT() && !falcon_got_lock) | ||||
| 		printk(KERN_ERR "scsi%d: !!BINGO!! Falcon has no lock in NCR5380_reset\n", | ||||
| 		       H_NO(cmd)); | ||||
| 
 | ||||
| 	NCR5380_print_status(cmd->device->host); | ||||
| 
 | ||||
| 	/* get in phase */ | ||||
|  |  | |||
|  | @ -184,7 +184,7 @@ static void atari_scsi_fetch_restbytes(void); | |||
| static irqreturn_t scsi_tt_intr(int irq, void *dummy); | ||||
| static irqreturn_t scsi_falcon_intr(int irq, void *dummy); | ||||
| static void falcon_release_lock_if_possible(struct NCR5380_hostdata *hostdata); | ||||
| static void falcon_get_lock(void); | ||||
| static int falcon_get_lock(void); | ||||
| #ifdef CONFIG_ATARI_SCSI_RESET_BOOT | ||||
| static void atari_scsi_reset_boot(void); | ||||
| #endif | ||||
|  | @ -473,17 +473,10 @@ static void atari_scsi_fetch_restbytes(void) | |||
| #endif /* REAL_DMA */ | ||||
| 
 | ||||
| 
 | ||||
| static int falcon_got_lock = 0; | ||||
| static DECLARE_WAIT_QUEUE_HEAD(falcon_fairness_wait); | ||||
| static int falcon_trying_lock = 0; | ||||
| static DECLARE_WAIT_QUEUE_HEAD(falcon_try_wait); | ||||
| static int falcon_dont_release = 0; | ||||
| 
 | ||||
| /* This function releases the lock on the DMA chip if there is no
 | ||||
|  * connected command and the disconnected queue is empty. On | ||||
|  * releasing, instances of falcon_get_lock are awoken, that put | ||||
|  * themselves to sleep for fairness. They can now try to get the lock | ||||
|  * again (but others waiting longer more probably will win). | ||||
|  * connected command and the disconnected queue is empty. | ||||
|  */ | ||||
| 
 | ||||
| static void falcon_release_lock_if_possible(struct NCR5380_hostdata *hostdata) | ||||
|  | @ -495,20 +488,12 @@ static void falcon_release_lock_if_possible(struct NCR5380_hostdata *hostdata) | |||
| 
 | ||||
| 	local_irq_save(flags); | ||||
| 
 | ||||
| 	if (falcon_got_lock && !hostdata->disconnected_queue && | ||||
| 	    !hostdata->issue_queue && !hostdata->connected) { | ||||
| 
 | ||||
| 		if (falcon_dont_release) { | ||||
| #if 0 | ||||
| 			printk("WARNING: Lock release not allowed. Ignored\n"); | ||||
| #endif | ||||
| 			local_irq_restore(flags); | ||||
| 			return; | ||||
| 		} | ||||
| 		falcon_got_lock = 0; | ||||
| 	if (!hostdata->disconnected_queue && | ||||
| 	    !hostdata->issue_queue && | ||||
| 	    !hostdata->connected && | ||||
| 	    !falcon_dont_release && | ||||
| 	    stdma_is_locked_by(scsi_falcon_intr)) | ||||
| 		stdma_release(); | ||||
| 		wake_up(&falcon_fairness_wait); | ||||
| 	} | ||||
| 
 | ||||
| 	local_irq_restore(flags); | ||||
| } | ||||
|  | @ -517,51 +502,19 @@ static void falcon_release_lock_if_possible(struct NCR5380_hostdata *hostdata) | |||
|  * If the DMA isn't locked already for SCSI, it tries to lock it by | ||||
|  * calling stdma_lock(). But if the DMA is locked by the SCSI code and | ||||
|  * there are other drivers waiting for the chip, we do not issue the | ||||
|  * command immediately but wait on 'falcon_fairness_queue'. We will be | ||||
|  * waked up when the DMA is unlocked by some SCSI interrupt. After that | ||||
|  * we try to get the lock again. | ||||
|  * But we must be prepared that more than one instance of | ||||
|  * falcon_get_lock() is waiting on the fairness queue. They should not | ||||
|  * try all at once to call stdma_lock(), one is enough! For that, the | ||||
|  * first one sets 'falcon_trying_lock', others that see that variable | ||||
|  * set wait on the queue 'falcon_try_wait'. | ||||
|  * Complicated, complicated.... Sigh... | ||||
|  * command immediately but tell the SCSI mid-layer to defer. | ||||
|  */ | ||||
| 
 | ||||
| static void falcon_get_lock(void) | ||||
| static int falcon_get_lock(void) | ||||
| { | ||||
| 	unsigned long flags; | ||||
| 
 | ||||
| 	if (IS_A_TT()) | ||||
| 		return; | ||||
| 		return 1; | ||||
| 
 | ||||
| 	local_irq_save(flags); | ||||
| 	if (in_interrupt()) | ||||
| 		return stdma_try_lock(scsi_falcon_intr, NULL); | ||||
| 
 | ||||
| 	wait_event_cmd(falcon_fairness_wait, | ||||
| 		in_interrupt() || !falcon_got_lock || !stdma_others_waiting(), | ||||
| 		local_irq_restore(flags), | ||||
| 		local_irq_save(flags)); | ||||
| 
 | ||||
| 	while (!falcon_got_lock) { | ||||
| 		if (in_irq()) | ||||
| 			panic("Falcon SCSI hasn't ST-DMA lock in interrupt"); | ||||
| 		if (!falcon_trying_lock) { | ||||
| 			falcon_trying_lock = 1; | ||||
| 			stdma_lock(scsi_falcon_intr, NULL); | ||||
| 			falcon_got_lock = 1; | ||||
| 			falcon_trying_lock = 0; | ||||
| 			wake_up(&falcon_try_wait); | ||||
| 		} else { | ||||
| 			wait_event_cmd(falcon_try_wait, | ||||
| 				falcon_got_lock && !falcon_trying_lock, | ||||
| 				local_irq_restore(flags), | ||||
| 				local_irq_save(flags)); | ||||
| 		} | ||||
| 	} | ||||
| 
 | ||||
| 	local_irq_restore(flags); | ||||
| 	if (!falcon_got_lock) | ||||
| 		panic("Falcon SCSI: someone stole the lock :-(\n"); | ||||
| 	stdma_lock(scsi_falcon_intr, NULL); | ||||
| 	return 1; | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
|  |  | |||
		Loading…
	
	Add table
		
		Reference in a new issue
	
	 Finn Thain
						Finn Thain