 5e8e6b4b3a
			
		
	
	
	5e8e6b4b3a
	
	
	
		
			
			This patch fixes a >= v3.10 regression bug with mutex_trylock() usage within iscsit_increment_maxcmdsn(), that was originally added to allow for a special case where ->cmdsn_mutex was already held from the iscsit_execute_cmd() exception path for ib_isert. When !mutex_trylock() was occuring under contention during normal RX/TX process context codepaths, the bug was manifesting itself as the following protocol error: Received CmdSN: 0x000fcbb7 is greater than MaxCmdSN: 0x000fcbb6, protocol error. Received CmdSN: 0x000fcbb8 is greater than MaxCmdSN: 0x000fcbb6, protocol error. This patch simply avoids the direct ib_isert callback in lio_queue_status() for the special iscsi_execute_cmd() exception cases, that allows the problematic mutex_trylock() usage in iscsit_increment_maxcmdsn() to go away. Reported-by: Moussa Ba <moussaba@micron.com> Tested-by: Moussa Ba <moussaba@micron.com> Cc: <stable@vger.kernel.org> # v3.10+ Signed-off-by: Nicholas Bellinger <nab@linux-iscsi.org>
		
			
				
	
	
		
			66 lines
		
	
	
	
		
			2.2 KiB
			
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			66 lines
		
	
	
	
		
			2.2 KiB
			
		
	
	
	
		
			C
		
	
	
	
	
	
| /*******************************************************************************
 | |
|  * This file contains the iSCSI Virtual Device and Disk Transport
 | |
|  * agnostic related functions.
 | |
|  *
 | |
|  * (c) Copyright 2007-2013 Datera, Inc.
 | |
|  *
 | |
|  * Author: Nicholas A. Bellinger <nab@linux-iscsi.org>
 | |
|  *
 | |
|  * This program is free software; you can redistribute it and/or modify
 | |
|  * it under the terms of the GNU General Public License as published by
 | |
|  * the Free Software Foundation; either version 2 of the License, or
 | |
|  * (at your option) any later version.
 | |
|  *
 | |
|  * This program is distributed in the hope that it will be useful,
 | |
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | |
|  * GNU General Public License for more details.
 | |
|  ******************************************************************************/
 | |
| 
 | |
| #include <scsi/scsi_device.h>
 | |
| #include <target/target_core_base.h>
 | |
| #include <target/target_core_fabric.h>
 | |
| 
 | |
| #include "iscsi_target_core.h"
 | |
| #include "iscsi_target_device.h"
 | |
| #include "iscsi_target_tpg.h"
 | |
| #include "iscsi_target_util.h"
 | |
| 
 | |
| void iscsit_determine_maxcmdsn(struct iscsi_session *sess)
 | |
| {
 | |
| 	struct se_node_acl *se_nacl;
 | |
| 
 | |
| 	/*
 | |
| 	 * This is a discovery session, the single queue slot was already
 | |
| 	 * assigned in iscsi_login_zero_tsih().  Since only Logout and
 | |
| 	 * Text Opcodes are allowed during discovery we do not have to worry
 | |
| 	 * about the HBA's queue depth here.
 | |
| 	 */
 | |
| 	if (sess->sess_ops->SessionType)
 | |
| 		return;
 | |
| 
 | |
| 	se_nacl = sess->se_sess->se_node_acl;
 | |
| 
 | |
| 	/*
 | |
| 	 * This is a normal session, set the Session's CmdSN window to the
 | |
| 	 * struct se_node_acl->queue_depth.  The value in struct se_node_acl->queue_depth
 | |
| 	 * has already been validated as a legal value in
 | |
| 	 * core_set_queue_depth_for_node().
 | |
| 	 */
 | |
| 	sess->cmdsn_window = se_nacl->queue_depth;
 | |
| 	sess->max_cmd_sn = (sess->max_cmd_sn + se_nacl->queue_depth) - 1;
 | |
| }
 | |
| 
 | |
| void iscsit_increment_maxcmdsn(struct iscsi_cmd *cmd, struct iscsi_session *sess)
 | |
| {
 | |
| 	if (cmd->immediate_cmd || cmd->maxcmdsn_inc)
 | |
| 		return;
 | |
| 
 | |
| 	cmd->maxcmdsn_inc = 1;
 | |
| 
 | |
| 	mutex_lock(&sess->cmdsn_mutex);
 | |
| 	sess->max_cmd_sn += 1;
 | |
| 	pr_debug("Updated MaxCmdSN to 0x%08x\n", sess->max_cmd_sn);
 | |
| 	mutex_unlock(&sess->cmdsn_mutex);
 | |
| }
 | |
| EXPORT_SYMBOL(iscsit_increment_maxcmdsn);
 |