mirror of
				https://github.com/torvalds/linux.git
				synced 2025-11-04 02:30:34 +02:00 
			
		
		
		
	target: Convert se_node_acl->device_list[] to RCU hlist
This patch converts se_node_acl->device_list[] table for mappedluns to modern RCU hlist_head usage in order to support an arbitrary number of node_acl lun mappings. It converts transport_lookup_*_lun() fast-path code to use RCU read path primitives when looking up se_dev_entry. It adds a new hlist_head at se_node_acl->lun_entry_hlist for this purpose. For transport_lookup_cmd_lun() code, it works with existing per-cpu se_lun->lun_ref when associating se_cmd with se_lun + se_device. Also, go ahead and update core_create_device_list_for_node() + core_free_device_list_for_node() to use ->lun_entry_hlist. It also converts se_dev_entry->pr_ref_count access to use modern struct kref counting, and updates core_disable_device_list_for_node() to kref_put() and block on se_deve->pr_comp waiting for outstanding PR special-case PR references to drop, then invoke kfree_rcu() to wait for the RCU grace period to complete before releasing memory. So now that se_node_acl->lun_entry_hlist fast path access uses RCU protected pointers, go ahead and convert remaining non-fast path RCU updater code using ->lun_entry_lock to struct mutex to allow callers to block while walking se_node_acl->lun_entry_hlist. Finally drop the left-over core_clear_initiator_node_from_tpg() that originally cleared lun_access during se_node_acl shutdown, as post RCU conversion it now becomes duplicated logic. Reviewed-by: Hannes Reinecke <hare@suse.de> Cc: Christoph Hellwig <hch@lst.de> Cc: Sagi Grimberg <sagig@mellanox.com> Cc: Paul E. McKenney <paulmck@linux.vnet.ibm.com> Signed-off-by: Nicholas Bellinger <nab@linux-iscsi.org>
This commit is contained in:
		
							parent
							
								
									d2c27f0d0b
								
							
						
					
					
						commit
						29a05deebf
					
				
					 12 changed files with 413 additions and 442 deletions
				
			
		| 
						 | 
				
			
			@ -1001,7 +1001,8 @@ static void core_alua_do_transition_tg_pt_work(struct work_struct *work)
 | 
			
		|||
		spin_lock_bh(&port->sep_alua_lock);
 | 
			
		||||
		list_for_each_entry(se_deve, &port->sep_alua_list,
 | 
			
		||||
					alua_port_list) {
 | 
			
		||||
			lacl = se_deve->se_lun_acl;
 | 
			
		||||
			lacl = rcu_dereference_check(se_deve->se_lun_acl,
 | 
			
		||||
					     lockdep_is_held(&port->sep_alua_lock));
 | 
			
		||||
			/*
 | 
			
		||||
			 * se_deve->se_lun_acl pointer may be NULL for a
 | 
			
		||||
			 * entry created without explicit Node+MappedLUN ACLs
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -60,18 +60,17 @@ transport_lookup_cmd_lun(struct se_cmd *se_cmd, u32 unpacked_lun)
 | 
			
		|||
{
 | 
			
		||||
	struct se_lun *se_lun = NULL;
 | 
			
		||||
	struct se_session *se_sess = se_cmd->se_sess;
 | 
			
		||||
	struct se_node_acl *nacl = se_sess->se_node_acl;
 | 
			
		||||
	struct se_device *dev;
 | 
			
		||||
	unsigned long flags;
 | 
			
		||||
	struct se_dev_entry *deve;
 | 
			
		||||
 | 
			
		||||
	if (unpacked_lun >= TRANSPORT_MAX_LUNS_PER_TPG)
 | 
			
		||||
		return TCM_NON_EXISTENT_LUN;
 | 
			
		||||
 | 
			
		||||
	spin_lock_irqsave(&se_sess->se_node_acl->device_list_lock, flags);
 | 
			
		||||
	se_cmd->se_deve = se_sess->se_node_acl->device_list[unpacked_lun];
 | 
			
		||||
	if (se_cmd->se_deve->lun_flags & TRANSPORT_LUNFLAGS_INITIATOR_ACCESS) {
 | 
			
		||||
		struct se_dev_entry *deve = se_cmd->se_deve;
 | 
			
		||||
 | 
			
		||||
		deve->total_cmds++;
 | 
			
		||||
	rcu_read_lock();
 | 
			
		||||
	deve = target_nacl_find_deve(nacl, unpacked_lun);
 | 
			
		||||
	if (deve) {
 | 
			
		||||
		atomic_long_inc(&deve->total_cmds);
 | 
			
		||||
 | 
			
		||||
		if ((se_cmd->data_direction == DMA_TO_DEVICE) &&
 | 
			
		||||
		    (deve->lun_flags & TRANSPORT_LUNFLAGS_READ_ONLY)) {
 | 
			
		||||
| 
						 | 
				
			
			@ -79,17 +78,19 @@ transport_lookup_cmd_lun(struct se_cmd *se_cmd, u32 unpacked_lun)
 | 
			
		|||
				" Access for 0x%08x\n",
 | 
			
		||||
				se_cmd->se_tfo->get_fabric_name(),
 | 
			
		||||
				unpacked_lun);
 | 
			
		||||
			spin_unlock_irqrestore(&se_sess->se_node_acl->device_list_lock, flags);
 | 
			
		||||
			rcu_read_unlock();
 | 
			
		||||
			return TCM_WRITE_PROTECTED;
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		if (se_cmd->data_direction == DMA_TO_DEVICE)
 | 
			
		||||
			deve->write_bytes += se_cmd->data_length;
 | 
			
		||||
			atomic_long_add(se_cmd->data_length,
 | 
			
		||||
					&deve->write_bytes);
 | 
			
		||||
		else if (se_cmd->data_direction == DMA_FROM_DEVICE)
 | 
			
		||||
			deve->read_bytes += se_cmd->data_length;
 | 
			
		||||
			atomic_long_add(se_cmd->data_length,
 | 
			
		||||
					&deve->read_bytes);
 | 
			
		||||
 | 
			
		||||
		se_lun = deve->se_lun;
 | 
			
		||||
		se_cmd->se_lun = deve->se_lun;
 | 
			
		||||
		se_lun = rcu_dereference(deve->se_lun);
 | 
			
		||||
		se_cmd->se_lun = rcu_dereference(deve->se_lun);
 | 
			
		||||
		se_cmd->pr_res_key = deve->pr_res_key;
 | 
			
		||||
		se_cmd->orig_fe_lun = unpacked_lun;
 | 
			
		||||
		se_cmd->se_cmd_flags |= SCF_SE_LUN_CMD;
 | 
			
		||||
| 
						 | 
				
			
			@ -97,7 +98,7 @@ transport_lookup_cmd_lun(struct se_cmd *se_cmd, u32 unpacked_lun)
 | 
			
		|||
		percpu_ref_get(&se_lun->lun_ref);
 | 
			
		||||
		se_cmd->lun_ref_active = true;
 | 
			
		||||
	}
 | 
			
		||||
	spin_unlock_irqrestore(&se_sess->se_node_acl->device_list_lock, flags);
 | 
			
		||||
	rcu_read_unlock();
 | 
			
		||||
 | 
			
		||||
	if (!se_lun) {
 | 
			
		||||
		/*
 | 
			
		||||
| 
						 | 
				
			
			@ -147,24 +148,23 @@ int transport_lookup_tmr_lun(struct se_cmd *se_cmd, u32 unpacked_lun)
 | 
			
		|||
	struct se_dev_entry *deve;
 | 
			
		||||
	struct se_lun *se_lun = NULL;
 | 
			
		||||
	struct se_session *se_sess = se_cmd->se_sess;
 | 
			
		||||
	struct se_node_acl *nacl = se_sess->se_node_acl;
 | 
			
		||||
	struct se_tmr_req *se_tmr = se_cmd->se_tmr_req;
 | 
			
		||||
	unsigned long flags;
 | 
			
		||||
 | 
			
		||||
	if (unpacked_lun >= TRANSPORT_MAX_LUNS_PER_TPG)
 | 
			
		||||
		return -ENODEV;
 | 
			
		||||
 | 
			
		||||
	spin_lock_irqsave(&se_sess->se_node_acl->device_list_lock, flags);
 | 
			
		||||
	se_cmd->se_deve = se_sess->se_node_acl->device_list[unpacked_lun];
 | 
			
		||||
	deve = se_cmd->se_deve;
 | 
			
		||||
 | 
			
		||||
	if (deve->lun_flags & TRANSPORT_LUNFLAGS_INITIATOR_ACCESS) {
 | 
			
		||||
		se_tmr->tmr_lun = deve->se_lun;
 | 
			
		||||
		se_cmd->se_lun = deve->se_lun;
 | 
			
		||||
		se_lun = deve->se_lun;
 | 
			
		||||
	rcu_read_lock();
 | 
			
		||||
	deve = target_nacl_find_deve(nacl, unpacked_lun);
 | 
			
		||||
	if (deve) {
 | 
			
		||||
		se_tmr->tmr_lun = rcu_dereference(deve->se_lun);
 | 
			
		||||
		se_cmd->se_lun = rcu_dereference(deve->se_lun);
 | 
			
		||||
		se_lun = rcu_dereference(deve->se_lun);
 | 
			
		||||
		se_cmd->pr_res_key = deve->pr_res_key;
 | 
			
		||||
		se_cmd->orig_fe_lun = unpacked_lun;
 | 
			
		||||
	}
 | 
			
		||||
	spin_unlock_irqrestore(&se_sess->se_node_acl->device_list_lock, flags);
 | 
			
		||||
	rcu_read_unlock();
 | 
			
		||||
 | 
			
		||||
	if (!se_lun) {
 | 
			
		||||
		pr_debug("TARGET_CORE[%s]: Detected NON_EXISTENT_LUN"
 | 
			
		||||
| 
						 | 
				
			
			@ -186,9 +186,27 @@ int transport_lookup_tmr_lun(struct se_cmd *se_cmd, u32 unpacked_lun)
 | 
			
		|||
}
 | 
			
		||||
EXPORT_SYMBOL(transport_lookup_tmr_lun);
 | 
			
		||||
 | 
			
		||||
bool target_lun_is_rdonly(struct se_cmd *cmd)
 | 
			
		||||
{
 | 
			
		||||
	struct se_session *se_sess = cmd->se_sess;
 | 
			
		||||
	struct se_dev_entry *deve;
 | 
			
		||||
	bool ret;
 | 
			
		||||
 | 
			
		||||
	if (cmd->se_lun->lun_access & TRANSPORT_LUNFLAGS_READ_ONLY)
 | 
			
		||||
		return true;
 | 
			
		||||
 | 
			
		||||
	rcu_read_lock();
 | 
			
		||||
	deve = target_nacl_find_deve(se_sess->se_node_acl, cmd->orig_fe_lun);
 | 
			
		||||
	ret = (deve && deve->lun_flags & TRANSPORT_LUNFLAGS_READ_ONLY);
 | 
			
		||||
	rcu_read_unlock();
 | 
			
		||||
 | 
			
		||||
	return ret;
 | 
			
		||||
}
 | 
			
		||||
EXPORT_SYMBOL(target_lun_is_rdonly);
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 * This function is called from core_scsi3_emulate_pro_register_and_move()
 | 
			
		||||
 * and core_scsi3_decode_spec_i_port(), and will increment &deve->pr_ref_count
 | 
			
		||||
 * and core_scsi3_decode_spec_i_port(), and will increment &deve->pr_kref
 | 
			
		||||
 * when a matching rtpi is found.
 | 
			
		||||
 */
 | 
			
		||||
struct se_dev_entry *core_get_se_deve_from_rtpi(
 | 
			
		||||
| 
						 | 
				
			
			@ -197,81 +215,43 @@ struct se_dev_entry *core_get_se_deve_from_rtpi(
 | 
			
		|||
{
 | 
			
		||||
	struct se_dev_entry *deve;
 | 
			
		||||
	struct se_lun *lun;
 | 
			
		||||
	struct se_port *port;
 | 
			
		||||
	struct se_portal_group *tpg = nacl->se_tpg;
 | 
			
		||||
	u32 i;
 | 
			
		||||
 | 
			
		||||
	spin_lock_irq(&nacl->device_list_lock);
 | 
			
		||||
	for (i = 0; i < TRANSPORT_MAX_LUNS_PER_TPG; i++) {
 | 
			
		||||
		deve = nacl->device_list[i];
 | 
			
		||||
 | 
			
		||||
		if (!(deve->lun_flags & TRANSPORT_LUNFLAGS_INITIATOR_ACCESS))
 | 
			
		||||
			continue;
 | 
			
		||||
 | 
			
		||||
		lun = deve->se_lun;
 | 
			
		||||
	rcu_read_lock();
 | 
			
		||||
	hlist_for_each_entry_rcu(deve, &nacl->lun_entry_hlist, link) {
 | 
			
		||||
		lun = rcu_dereference(deve->se_lun);
 | 
			
		||||
		if (!lun) {
 | 
			
		||||
			pr_err("%s device entries device pointer is"
 | 
			
		||||
				" NULL, but Initiator has access.\n",
 | 
			
		||||
				tpg->se_tpg_tfo->get_fabric_name());
 | 
			
		||||
			continue;
 | 
			
		||||
		}
 | 
			
		||||
		port = lun->lun_sep;
 | 
			
		||||
		if (!port) {
 | 
			
		||||
			pr_err("%s device entries device pointer is"
 | 
			
		||||
				" NULL, but Initiator has access.\n",
 | 
			
		||||
				tpg->se_tpg_tfo->get_fabric_name());
 | 
			
		||||
			continue;
 | 
			
		||||
		}
 | 
			
		||||
		if (port->sep_rtpi != rtpi)
 | 
			
		||||
		if (lun->lun_rtpi != rtpi)
 | 
			
		||||
			continue;
 | 
			
		||||
 | 
			
		||||
		atomic_inc_mb(&deve->pr_ref_count);
 | 
			
		||||
		spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
		kref_get(&deve->pr_kref);
 | 
			
		||||
		rcu_read_unlock();
 | 
			
		||||
 | 
			
		||||
		return deve;
 | 
			
		||||
	}
 | 
			
		||||
	spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
	rcu_read_unlock();
 | 
			
		||||
 | 
			
		||||
	return NULL;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
int core_free_device_list_for_node(
 | 
			
		||||
void core_free_device_list_for_node(
 | 
			
		||||
	struct se_node_acl *nacl,
 | 
			
		||||
	struct se_portal_group *tpg)
 | 
			
		||||
{
 | 
			
		||||
	struct se_dev_entry *deve;
 | 
			
		||||
	struct se_lun *lun;
 | 
			
		||||
	u32 i;
 | 
			
		||||
 | 
			
		||||
	if (!nacl->device_list)
 | 
			
		||||
		return 0;
 | 
			
		||||
 | 
			
		||||
	spin_lock_irq(&nacl->device_list_lock);
 | 
			
		||||
	for (i = 0; i < TRANSPORT_MAX_LUNS_PER_TPG; i++) {
 | 
			
		||||
		deve = nacl->device_list[i];
 | 
			
		||||
 | 
			
		||||
		if (!(deve->lun_flags & TRANSPORT_LUNFLAGS_INITIATOR_ACCESS))
 | 
			
		||||
			continue;
 | 
			
		||||
 | 
			
		||||
		if (!deve->se_lun) {
 | 
			
		||||
			pr_err("%s device entries device pointer is"
 | 
			
		||||
				" NULL, but Initiator has access.\n",
 | 
			
		||||
				tpg->se_tpg_tfo->get_fabric_name());
 | 
			
		||||
			continue;
 | 
			
		||||
		}
 | 
			
		||||
		lun = deve->se_lun;
 | 
			
		||||
 | 
			
		||||
		spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
		core_disable_device_list_for_node(lun, NULL, deve->mapped_lun,
 | 
			
		||||
			TRANSPORT_LUNFLAGS_NO_ACCESS, nacl, tpg);
 | 
			
		||||
		spin_lock_irq(&nacl->device_list_lock);
 | 
			
		||||
	mutex_lock(&nacl->lun_entry_mutex);
 | 
			
		||||
	hlist_for_each_entry_rcu(deve, &nacl->lun_entry_hlist, link) {
 | 
			
		||||
		struct se_lun *lun = rcu_dereference_check(deve->se_lun,
 | 
			
		||||
					lockdep_is_held(&nacl->lun_entry_mutex));
 | 
			
		||||
		core_disable_device_list_for_node(lun, deve, nacl, tpg);
 | 
			
		||||
	}
 | 
			
		||||
	spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
 | 
			
		||||
	array_free(nacl->device_list, TRANSPORT_MAX_LUNS_PER_TPG);
 | 
			
		||||
	nacl->device_list = NULL;
 | 
			
		||||
 | 
			
		||||
	return 0;
 | 
			
		||||
	mutex_unlock(&nacl->lun_entry_mutex);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void core_update_device_list_access(
 | 
			
		||||
| 
						 | 
				
			
			@ -281,16 +261,40 @@ void core_update_device_list_access(
 | 
			
		|||
{
 | 
			
		||||
	struct se_dev_entry *deve;
 | 
			
		||||
 | 
			
		||||
	spin_lock_irq(&nacl->device_list_lock);
 | 
			
		||||
	deve = nacl->device_list[mapped_lun];
 | 
			
		||||
	if (lun_access & TRANSPORT_LUNFLAGS_READ_WRITE) {
 | 
			
		||||
		deve->lun_flags &= ~TRANSPORT_LUNFLAGS_READ_ONLY;
 | 
			
		||||
		deve->lun_flags |= TRANSPORT_LUNFLAGS_READ_WRITE;
 | 
			
		||||
	} else {
 | 
			
		||||
		deve->lun_flags &= ~TRANSPORT_LUNFLAGS_READ_WRITE;
 | 
			
		||||
		deve->lun_flags |= TRANSPORT_LUNFLAGS_READ_ONLY;
 | 
			
		||||
	mutex_lock(&nacl->lun_entry_mutex);
 | 
			
		||||
	deve = target_nacl_find_deve(nacl, mapped_lun);
 | 
			
		||||
	if (deve) {
 | 
			
		||||
		if (lun_access & TRANSPORT_LUNFLAGS_READ_WRITE) {
 | 
			
		||||
			deve->lun_flags &= ~TRANSPORT_LUNFLAGS_READ_ONLY;
 | 
			
		||||
			deve->lun_flags |= TRANSPORT_LUNFLAGS_READ_WRITE;
 | 
			
		||||
		} else {
 | 
			
		||||
			deve->lun_flags &= ~TRANSPORT_LUNFLAGS_READ_WRITE;
 | 
			
		||||
			deve->lun_flags |= TRANSPORT_LUNFLAGS_READ_ONLY;
 | 
			
		||||
		}
 | 
			
		||||
	}
 | 
			
		||||
	spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
	mutex_unlock(&nacl->lun_entry_mutex);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 * Called with rcu_read_lock or nacl->device_list_lock held.
 | 
			
		||||
 */
 | 
			
		||||
struct se_dev_entry *target_nacl_find_deve(struct se_node_acl *nacl, u32 mapped_lun)
 | 
			
		||||
{
 | 
			
		||||
	struct se_dev_entry *deve;
 | 
			
		||||
 | 
			
		||||
	hlist_for_each_entry_rcu(deve, &nacl->lun_entry_hlist, link)
 | 
			
		||||
		if (deve->mapped_lun == mapped_lun)
 | 
			
		||||
			return deve;
 | 
			
		||||
 | 
			
		||||
	return NULL;
 | 
			
		||||
}
 | 
			
		||||
EXPORT_SYMBOL(target_nacl_find_deve);
 | 
			
		||||
 | 
			
		||||
void target_pr_kref_release(struct kref *kref)
 | 
			
		||||
{
 | 
			
		||||
	struct se_dev_entry *deve = container_of(kref, struct se_dev_entry,
 | 
			
		||||
						 pr_kref);
 | 
			
		||||
	complete(&deve->pr_comp);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/*      core_enable_device_list_for_node():
 | 
			
		||||
| 
						 | 
				
			
			@ -306,85 +310,87 @@ int core_enable_device_list_for_node(
 | 
			
		|||
	struct se_portal_group *tpg)
 | 
			
		||||
{
 | 
			
		||||
	struct se_port *port = lun->lun_sep;
 | 
			
		||||
	struct se_dev_entry *deve;
 | 
			
		||||
	struct se_dev_entry *orig, *new;
 | 
			
		||||
 | 
			
		||||
	spin_lock_irq(&nacl->device_list_lock);
 | 
			
		||||
	new = kzalloc(sizeof(*new), GFP_KERNEL);
 | 
			
		||||
	if (!new) {
 | 
			
		||||
		pr_err("Unable to allocate se_dev_entry memory\n");
 | 
			
		||||
		return -ENOMEM;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	deve = nacl->device_list[mapped_lun];
 | 
			
		||||
	atomic_set(&new->ua_count, 0);
 | 
			
		||||
	spin_lock_init(&new->ua_lock);
 | 
			
		||||
	INIT_LIST_HEAD(&new->alua_port_list);
 | 
			
		||||
	INIT_LIST_HEAD(&new->ua_list);
 | 
			
		||||
 | 
			
		||||
	/*
 | 
			
		||||
	 * Check if the call is handling demo mode -> explicit LUN ACL
 | 
			
		||||
	 * transition.  This transition must be for the same struct se_lun
 | 
			
		||||
	 * + mapped_lun that was setup in demo mode..
 | 
			
		||||
	 */
 | 
			
		||||
	if (deve->lun_flags & TRANSPORT_LUNFLAGS_INITIATOR_ACCESS) {
 | 
			
		||||
		if (deve->se_lun_acl != NULL) {
 | 
			
		||||
			pr_err("struct se_dev_entry->se_lun_acl"
 | 
			
		||||
			       " already set for demo mode -> explicit"
 | 
			
		||||
			       " LUN ACL transition\n");
 | 
			
		||||
			spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
	new->mapped_lun = mapped_lun;
 | 
			
		||||
	kref_init(&new->pr_kref);
 | 
			
		||||
	init_completion(&new->pr_comp);
 | 
			
		||||
 | 
			
		||||
	if (lun_access & TRANSPORT_LUNFLAGS_READ_WRITE)
 | 
			
		||||
		new->lun_flags |= TRANSPORT_LUNFLAGS_READ_WRITE;
 | 
			
		||||
	else
 | 
			
		||||
		new->lun_flags |= TRANSPORT_LUNFLAGS_READ_ONLY;
 | 
			
		||||
 | 
			
		||||
	new->creation_time = get_jiffies_64();
 | 
			
		||||
	new->attach_count++;
 | 
			
		||||
 | 
			
		||||
	mutex_lock(&nacl->lun_entry_mutex);
 | 
			
		||||
	orig = target_nacl_find_deve(nacl, mapped_lun);
 | 
			
		||||
	if (orig && orig->se_lun) {
 | 
			
		||||
		struct se_lun *orig_lun = rcu_dereference_check(orig->se_lun,
 | 
			
		||||
					lockdep_is_held(&nacl->lun_entry_mutex));
 | 
			
		||||
 | 
			
		||||
		if (orig_lun != lun) {
 | 
			
		||||
			pr_err("Existing orig->se_lun doesn't match new lun"
 | 
			
		||||
			       " for dynamic -> explicit NodeACL conversion:"
 | 
			
		||||
				" %s\n", nacl->initiatorname);
 | 
			
		||||
			mutex_unlock(&nacl->lun_entry_mutex);
 | 
			
		||||
			kfree(new);
 | 
			
		||||
			return -EINVAL;
 | 
			
		||||
		}
 | 
			
		||||
		if (deve->se_lun != lun) {
 | 
			
		||||
			pr_err("struct se_dev_entry->se_lun does"
 | 
			
		||||
			       " match passed struct se_lun for demo mode"
 | 
			
		||||
			       " -> explicit LUN ACL transition\n");
 | 
			
		||||
			spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
			return -EINVAL;
 | 
			
		||||
		}
 | 
			
		||||
		deve->se_lun_acl = lun_acl;
 | 
			
		||||
		BUG_ON(orig->se_lun_acl != NULL);
 | 
			
		||||
 | 
			
		||||
		if (lun_access & TRANSPORT_LUNFLAGS_READ_WRITE) {
 | 
			
		||||
			deve->lun_flags &= ~TRANSPORT_LUNFLAGS_READ_ONLY;
 | 
			
		||||
			deve->lun_flags |= TRANSPORT_LUNFLAGS_READ_WRITE;
 | 
			
		||||
		} else {
 | 
			
		||||
			deve->lun_flags &= ~TRANSPORT_LUNFLAGS_READ_WRITE;
 | 
			
		||||
			deve->lun_flags |= TRANSPORT_LUNFLAGS_READ_ONLY;
 | 
			
		||||
		}
 | 
			
		||||
		rcu_assign_pointer(new->se_lun, lun);
 | 
			
		||||
		rcu_assign_pointer(new->se_lun_acl, lun_acl);
 | 
			
		||||
		hlist_del_rcu(&orig->link);
 | 
			
		||||
		hlist_add_head_rcu(&new->link, &nacl->lun_entry_hlist);
 | 
			
		||||
		mutex_unlock(&nacl->lun_entry_mutex);
 | 
			
		||||
 | 
			
		||||
		spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
		spin_lock_bh(&port->sep_alua_lock);
 | 
			
		||||
		list_del(&orig->alua_port_list);
 | 
			
		||||
		list_add_tail(&new->alua_port_list, &port->sep_alua_list);
 | 
			
		||||
		spin_unlock_bh(&port->sep_alua_lock);
 | 
			
		||||
 | 
			
		||||
		kref_put(&orig->pr_kref, target_pr_kref_release);
 | 
			
		||||
		wait_for_completion(&orig->pr_comp);
 | 
			
		||||
 | 
			
		||||
		kfree_rcu(orig, rcu_head);
 | 
			
		||||
		return 0;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	deve->se_lun = lun;
 | 
			
		||||
	deve->se_lun_acl = lun_acl;
 | 
			
		||||
	deve->mapped_lun = mapped_lun;
 | 
			
		||||
	deve->lun_flags |= TRANSPORT_LUNFLAGS_INITIATOR_ACCESS;
 | 
			
		||||
 | 
			
		||||
	if (lun_access & TRANSPORT_LUNFLAGS_READ_WRITE) {
 | 
			
		||||
		deve->lun_flags &= ~TRANSPORT_LUNFLAGS_READ_ONLY;
 | 
			
		||||
		deve->lun_flags |= TRANSPORT_LUNFLAGS_READ_WRITE;
 | 
			
		||||
	} else {
 | 
			
		||||
		deve->lun_flags &= ~TRANSPORT_LUNFLAGS_READ_WRITE;
 | 
			
		||||
		deve->lun_flags |= TRANSPORT_LUNFLAGS_READ_ONLY;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	deve->creation_time = get_jiffies_64();
 | 
			
		||||
	deve->attach_count++;
 | 
			
		||||
	spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
	rcu_assign_pointer(new->se_lun, lun);
 | 
			
		||||
	rcu_assign_pointer(new->se_lun_acl, lun_acl);
 | 
			
		||||
	hlist_add_head_rcu(&new->link, &nacl->lun_entry_hlist);
 | 
			
		||||
	mutex_unlock(&nacl->lun_entry_mutex);
 | 
			
		||||
 | 
			
		||||
	spin_lock_bh(&port->sep_alua_lock);
 | 
			
		||||
	list_add_tail(&deve->alua_port_list, &port->sep_alua_list);
 | 
			
		||||
	list_add_tail(&new->alua_port_list, &port->sep_alua_list);
 | 
			
		||||
	spin_unlock_bh(&port->sep_alua_lock);
 | 
			
		||||
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/*      core_disable_device_list_for_node():
 | 
			
		||||
 *
 | 
			
		||||
 *
 | 
			
		||||
/*
 | 
			
		||||
 *	Called with se_node_acl->lun_entry_mutex held.
 | 
			
		||||
 */
 | 
			
		||||
int core_disable_device_list_for_node(
 | 
			
		||||
void core_disable_device_list_for_node(
 | 
			
		||||
	struct se_lun *lun,
 | 
			
		||||
	struct se_lun_acl *lun_acl,
 | 
			
		||||
	u32 mapped_lun,
 | 
			
		||||
	u32 lun_access,
 | 
			
		||||
	struct se_dev_entry *orig,
 | 
			
		||||
	struct se_node_acl *nacl,
 | 
			
		||||
	struct se_portal_group *tpg)
 | 
			
		||||
{
 | 
			
		||||
	struct se_port *port = lun->lun_sep;
 | 
			
		||||
	struct se_dev_entry *deve = nacl->device_list[mapped_lun];
 | 
			
		||||
 | 
			
		||||
	/*
 | 
			
		||||
	 * If the MappedLUN entry is being disabled, the entry in
 | 
			
		||||
	 * port->sep_alua_list must be removed now before clearing the
 | 
			
		||||
| 
						 | 
				
			
			@ -399,29 +405,29 @@ int core_disable_device_list_for_node(
 | 
			
		|||
	 * MappedLUN *deve will be released below..
 | 
			
		||||
	 */
 | 
			
		||||
	spin_lock_bh(&port->sep_alua_lock);
 | 
			
		||||
	list_del(&deve->alua_port_list);
 | 
			
		||||
	list_del(&orig->alua_port_list);
 | 
			
		||||
	spin_unlock_bh(&port->sep_alua_lock);
 | 
			
		||||
	/*
 | 
			
		||||
	 * Wait for any in process SPEC_I_PT=1 or REGISTER_AND_MOVE
 | 
			
		||||
	 * PR operation to complete.
 | 
			
		||||
	 */
 | 
			
		||||
	while (atomic_read(&deve->pr_ref_count) != 0)
 | 
			
		||||
		cpu_relax();
 | 
			
		||||
 | 
			
		||||
	spin_lock_irq(&nacl->device_list_lock);
 | 
			
		||||
	/*
 | 
			
		||||
	 * Disable struct se_dev_entry LUN ACL mapping
 | 
			
		||||
	 */
 | 
			
		||||
	core_scsi3_ua_release_all(deve);
 | 
			
		||||
	deve->se_lun = NULL;
 | 
			
		||||
	deve->se_lun_acl = NULL;
 | 
			
		||||
	deve->lun_flags = 0;
 | 
			
		||||
	deve->creation_time = 0;
 | 
			
		||||
	deve->attach_count--;
 | 
			
		||||
	spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
	core_scsi3_ua_release_all(orig);
 | 
			
		||||
 | 
			
		||||
	hlist_del_rcu(&orig->link);
 | 
			
		||||
	rcu_assign_pointer(orig->se_lun, NULL);
 | 
			
		||||
	rcu_assign_pointer(orig->se_lun_acl, NULL);
 | 
			
		||||
	orig->lun_flags = 0;
 | 
			
		||||
	orig->creation_time = 0;
 | 
			
		||||
	orig->attach_count--;
 | 
			
		||||
	/*
 | 
			
		||||
	 * Before firing off RCU callback, wait for any in process SPEC_I_PT=1
 | 
			
		||||
	 * or REGISTER_AND_MOVE PR operation to complete.
 | 
			
		||||
	 */
 | 
			
		||||
	kref_put(&orig->pr_kref, target_pr_kref_release);
 | 
			
		||||
	wait_for_completion(&orig->pr_comp);
 | 
			
		||||
 | 
			
		||||
	kfree_rcu(orig, rcu_head);
 | 
			
		||||
 | 
			
		||||
	core_scsi3_free_pr_reg_from_nacl(lun->lun_se_dev, nacl);
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/*      core_clear_lun_from_tpg():
 | 
			
		||||
| 
						 | 
				
			
			@ -432,26 +438,22 @@ void core_clear_lun_from_tpg(struct se_lun *lun, struct se_portal_group *tpg)
 | 
			
		|||
{
 | 
			
		||||
	struct se_node_acl *nacl;
 | 
			
		||||
	struct se_dev_entry *deve;
 | 
			
		||||
	u32 i;
 | 
			
		||||
 | 
			
		||||
	spin_lock_irq(&tpg->acl_node_lock);
 | 
			
		||||
	list_for_each_entry(nacl, &tpg->acl_node_list, acl_list) {
 | 
			
		||||
		spin_unlock_irq(&tpg->acl_node_lock);
 | 
			
		||||
 | 
			
		||||
		spin_lock_irq(&nacl->device_list_lock);
 | 
			
		||||
		for (i = 0; i < TRANSPORT_MAX_LUNS_PER_TPG; i++) {
 | 
			
		||||
			deve = nacl->device_list[i];
 | 
			
		||||
			if (lun != deve->se_lun)
 | 
			
		||||
		mutex_lock(&nacl->lun_entry_mutex);
 | 
			
		||||
		hlist_for_each_entry_rcu(deve, &nacl->lun_entry_hlist, link) {
 | 
			
		||||
			struct se_lun *tmp_lun = rcu_dereference_check(deve->se_lun,
 | 
			
		||||
					lockdep_is_held(&nacl->lun_entry_mutex));
 | 
			
		||||
 | 
			
		||||
			if (lun != tmp_lun)
 | 
			
		||||
				continue;
 | 
			
		||||
			spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
 | 
			
		||||
			core_disable_device_list_for_node(lun, NULL,
 | 
			
		||||
				deve->mapped_lun, TRANSPORT_LUNFLAGS_NO_ACCESS,
 | 
			
		||||
				nacl, tpg);
 | 
			
		||||
 | 
			
		||||
			spin_lock_irq(&nacl->device_list_lock);
 | 
			
		||||
			core_disable_device_list_for_node(lun, deve, nacl, tpg);
 | 
			
		||||
		}
 | 
			
		||||
		spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
		mutex_unlock(&nacl->lun_entry_mutex);
 | 
			
		||||
 | 
			
		||||
		spin_lock_irq(&tpg->acl_node_lock);
 | 
			
		||||
	}
 | 
			
		||||
| 
						 | 
				
			
			@ -583,7 +585,9 @@ int core_dev_export(
 | 
			
		|||
	if (IS_ERR(port))
 | 
			
		||||
		return PTR_ERR(port);
 | 
			
		||||
 | 
			
		||||
	lun->lun_index = dev->dev_index;
 | 
			
		||||
	lun->lun_se_dev = dev;
 | 
			
		||||
	lun->lun_rtpi = port->sep_rtpi;
 | 
			
		||||
 | 
			
		||||
	spin_lock(&hba->device_lock);
 | 
			
		||||
	dev->export_count++;
 | 
			
		||||
| 
						 | 
				
			
			@ -1369,16 +1373,13 @@ int core_dev_add_initiator_node_lun_acl(
 | 
			
		|||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/*      core_dev_del_initiator_node_lun_acl():
 | 
			
		||||
 *
 | 
			
		||||
 *
 | 
			
		||||
 */
 | 
			
		||||
int core_dev_del_initiator_node_lun_acl(
 | 
			
		||||
	struct se_portal_group *tpg,
 | 
			
		||||
	struct se_lun *lun,
 | 
			
		||||
	struct se_lun_acl *lacl)
 | 
			
		||||
{
 | 
			
		||||
	struct se_node_acl *nacl;
 | 
			
		||||
	struct se_dev_entry *deve;
 | 
			
		||||
 | 
			
		||||
	nacl = lacl->se_lun_nacl;
 | 
			
		||||
	if (!nacl)
 | 
			
		||||
| 
						 | 
				
			
			@ -1389,8 +1390,11 @@ int core_dev_del_initiator_node_lun_acl(
 | 
			
		|||
	atomic_dec_mb(&lun->lun_acl_count);
 | 
			
		||||
	spin_unlock(&lun->lun_acl_lock);
 | 
			
		||||
 | 
			
		||||
	core_disable_device_list_for_node(lun, NULL, lacl->mapped_lun,
 | 
			
		||||
		TRANSPORT_LUNFLAGS_NO_ACCESS, nacl, tpg);
 | 
			
		||||
	mutex_lock(&nacl->lun_entry_mutex);
 | 
			
		||||
	deve = target_nacl_find_deve(nacl, lacl->mapped_lun);
 | 
			
		||||
	if (deve)
 | 
			
		||||
		core_disable_device_list_for_node(lun, deve, nacl, tpg);
 | 
			
		||||
	mutex_unlock(&nacl->lun_entry_mutex);
 | 
			
		||||
 | 
			
		||||
	lacl->se_lun = NULL;
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -123,16 +123,16 @@ static int target_fabric_mappedlun_link(
 | 
			
		|||
	 * which be will write protected (READ-ONLY) when
 | 
			
		||||
	 * tpg_1/attrib/demo_mode_write_protect=1
 | 
			
		||||
	 */
 | 
			
		||||
	spin_lock_irq(&lacl->se_lun_nacl->device_list_lock);
 | 
			
		||||
	deve = lacl->se_lun_nacl->device_list[lacl->mapped_lun];
 | 
			
		||||
	if (deve->lun_flags & TRANSPORT_LUNFLAGS_INITIATOR_ACCESS)
 | 
			
		||||
	rcu_read_lock();
 | 
			
		||||
	deve = target_nacl_find_deve(lacl->se_lun_nacl, lacl->mapped_lun);
 | 
			
		||||
	if (deve)
 | 
			
		||||
		lun_access = deve->lun_flags;
 | 
			
		||||
	else
 | 
			
		||||
		lun_access =
 | 
			
		||||
			(se_tpg->se_tpg_tfo->tpg_check_prod_mode_write_protect(
 | 
			
		||||
				se_tpg)) ? TRANSPORT_LUNFLAGS_READ_ONLY :
 | 
			
		||||
					   TRANSPORT_LUNFLAGS_READ_WRITE;
 | 
			
		||||
	spin_unlock_irq(&lacl->se_lun_nacl->device_list_lock);
 | 
			
		||||
	rcu_read_unlock();
 | 
			
		||||
	/*
 | 
			
		||||
	 * Determine the actual mapped LUN value user wants..
 | 
			
		||||
	 *
 | 
			
		||||
| 
						 | 
				
			
			@ -149,23 +149,13 @@ static int target_fabric_mappedlun_unlink(
 | 
			
		|||
	struct config_item *lun_acl_ci,
 | 
			
		||||
	struct config_item *lun_ci)
 | 
			
		||||
{
 | 
			
		||||
	struct se_lun *lun;
 | 
			
		||||
	struct se_lun_acl *lacl = container_of(to_config_group(lun_acl_ci),
 | 
			
		||||
			struct se_lun_acl, se_lun_group);
 | 
			
		||||
	struct se_node_acl *nacl = lacl->se_lun_nacl;
 | 
			
		||||
	struct se_dev_entry *deve = nacl->device_list[lacl->mapped_lun];
 | 
			
		||||
	struct se_portal_group *se_tpg;
 | 
			
		||||
	/*
 | 
			
		||||
	 * Determine if the underlying MappedLUN has already been released..
 | 
			
		||||
	 */
 | 
			
		||||
	if (!deve->se_lun)
 | 
			
		||||
		return 0;
 | 
			
		||||
	struct se_lun *lun = container_of(to_config_group(lun_ci),
 | 
			
		||||
			struct se_lun, lun_group);
 | 
			
		||||
	struct se_portal_group *se_tpg = lun->lun_sep->sep_tpg;
 | 
			
		||||
 | 
			
		||||
	lun = container_of(to_config_group(lun_ci), struct se_lun, lun_group);
 | 
			
		||||
	se_tpg = lun->lun_sep->sep_tpg;
 | 
			
		||||
 | 
			
		||||
	core_dev_del_initiator_node_lun_acl(se_tpg, lun, lacl);
 | 
			
		||||
	return 0;
 | 
			
		||||
	return core_dev_del_initiator_node_lun_acl(se_tpg, lun, lacl);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
CONFIGFS_EATTR_STRUCT(target_fabric_mappedlun, se_lun_acl);
 | 
			
		||||
| 
						 | 
				
			
			@ -181,14 +171,15 @@ static ssize_t target_fabric_mappedlun_show_write_protect(
 | 
			
		|||
{
 | 
			
		||||
	struct se_node_acl *se_nacl = lacl->se_lun_nacl;
 | 
			
		||||
	struct se_dev_entry *deve;
 | 
			
		||||
	ssize_t len;
 | 
			
		||||
	ssize_t len = 0;
 | 
			
		||||
 | 
			
		||||
	spin_lock_irq(&se_nacl->device_list_lock);
 | 
			
		||||
	deve = se_nacl->device_list[lacl->mapped_lun];
 | 
			
		||||
	len = sprintf(page, "%d\n",
 | 
			
		||||
			(deve->lun_flags & TRANSPORT_LUNFLAGS_READ_ONLY) ?
 | 
			
		||||
			1 : 0);
 | 
			
		||||
	spin_unlock_irq(&se_nacl->device_list_lock);
 | 
			
		||||
	rcu_read_lock();
 | 
			
		||||
	deve = target_nacl_find_deve(se_nacl, lacl->mapped_lun);
 | 
			
		||||
	if (deve) {
 | 
			
		||||
		len = sprintf(page, "%d\n",
 | 
			
		||||
			(deve->lun_flags & TRANSPORT_LUNFLAGS_READ_ONLY) ? 1 : 0);
 | 
			
		||||
	}
 | 
			
		||||
	rcu_read_unlock();
 | 
			
		||||
 | 
			
		||||
	return len;
 | 
			
		||||
}
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -9,13 +9,15 @@ extern struct mutex g_device_mutex;
 | 
			
		|||
extern struct list_head g_device_list;
 | 
			
		||||
 | 
			
		||||
struct se_dev_entry *core_get_se_deve_from_rtpi(struct se_node_acl *, u16);
 | 
			
		||||
int	core_free_device_list_for_node(struct se_node_acl *,
 | 
			
		||||
void	target_pr_kref_release(struct kref *);
 | 
			
		||||
void	core_free_device_list_for_node(struct se_node_acl *,
 | 
			
		||||
		struct se_portal_group *);
 | 
			
		||||
void	core_update_device_list_access(u32, u32, struct se_node_acl *);
 | 
			
		||||
struct se_dev_entry *target_nacl_find_deve(struct se_node_acl *, u32);
 | 
			
		||||
int	core_enable_device_list_for_node(struct se_lun *, struct se_lun_acl *,
 | 
			
		||||
		u32, u32, struct se_node_acl *, struct se_portal_group *);
 | 
			
		||||
int	core_disable_device_list_for_node(struct se_lun *, struct se_lun_acl *,
 | 
			
		||||
		u32, u32, struct se_node_acl *, struct se_portal_group *);
 | 
			
		||||
void	core_disable_device_list_for_node(struct se_lun *, struct se_dev_entry *,
 | 
			
		||||
		struct se_node_acl *, struct se_portal_group *);
 | 
			
		||||
void	core_clear_lun_from_tpg(struct se_lun *, struct se_portal_group *);
 | 
			
		||||
int	core_dev_export(struct se_device *, struct se_portal_group *,
 | 
			
		||||
		struct se_lun *);
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -48,7 +48,7 @@ struct pr_transport_id_holder {
 | 
			
		|||
	struct t10_pr_registration *dest_pr_reg;
 | 
			
		||||
	struct se_portal_group *dest_tpg;
 | 
			
		||||
	struct se_node_acl *dest_node_acl;
 | 
			
		||||
	struct se_dev_entry *dest_se_deve;
 | 
			
		||||
	struct se_dev_entry __rcu *dest_se_deve;
 | 
			
		||||
	struct list_head dest_list;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -232,7 +232,7 @@ target_scsi2_reservation_release(struct se_cmd *cmd)
 | 
			
		|||
	tpg = sess->se_tpg;
 | 
			
		||||
	pr_debug("SCSI-2 Released reservation for %s LUN: %u ->"
 | 
			
		||||
		" MAPPED LUN: %u for %s\n", tpg->se_tpg_tfo->get_fabric_name(),
 | 
			
		||||
		cmd->se_lun->unpacked_lun, cmd->se_deve->mapped_lun,
 | 
			
		||||
		cmd->se_lun->unpacked_lun, cmd->orig_fe_lun,
 | 
			
		||||
		sess->se_node_acl->initiatorname);
 | 
			
		||||
 | 
			
		||||
out_unlock:
 | 
			
		||||
| 
						 | 
				
			
			@ -281,7 +281,7 @@ target_scsi2_reservation_reserve(struct se_cmd *cmd)
 | 
			
		|||
			dev->dev_reserved_node_acl->initiatorname);
 | 
			
		||||
		pr_err("Current attempt - LUN: %u -> MAPPED LUN: %u"
 | 
			
		||||
			" from %s \n", cmd->se_lun->unpacked_lun,
 | 
			
		||||
			cmd->se_deve->mapped_lun,
 | 
			
		||||
			cmd->orig_fe_lun,
 | 
			
		||||
			sess->se_node_acl->initiatorname);
 | 
			
		||||
		ret = TCM_RESERVATION_CONFLICT;
 | 
			
		||||
		goto out_unlock;
 | 
			
		||||
| 
						 | 
				
			
			@ -295,7 +295,7 @@ target_scsi2_reservation_reserve(struct se_cmd *cmd)
 | 
			
		|||
	}
 | 
			
		||||
	pr_debug("SCSI-2 Reserved %s LUN: %u -> MAPPED LUN: %u"
 | 
			
		||||
		" for %s\n", tpg->se_tpg_tfo->get_fabric_name(),
 | 
			
		||||
		cmd->se_lun->unpacked_lun, cmd->se_deve->mapped_lun,
 | 
			
		||||
		cmd->se_lun->unpacked_lun, cmd->orig_fe_lun,
 | 
			
		||||
		sess->se_node_acl->initiatorname);
 | 
			
		||||
 | 
			
		||||
out_unlock:
 | 
			
		||||
| 
						 | 
				
			
			@ -320,6 +320,7 @@ static int core_scsi3_pr_seq_non_holder(
 | 
			
		|||
	unsigned char *cdb = cmd->t_task_cdb;
 | 
			
		||||
	struct se_dev_entry *se_deve;
 | 
			
		||||
	struct se_session *se_sess = cmd->se_sess;
 | 
			
		||||
	struct se_node_acl *nacl = se_sess->se_node_acl;
 | 
			
		||||
	int other_cdb = 0, ignore_reg;
 | 
			
		||||
	int registered_nexus = 0, ret = 1; /* Conflict by default */
 | 
			
		||||
	int all_reg = 0, reg_only = 0; /* ALL_REG, REG_ONLY */
 | 
			
		||||
| 
						 | 
				
			
			@ -327,7 +328,8 @@ static int core_scsi3_pr_seq_non_holder(
 | 
			
		|||
	int legacy = 0; /* Act like a legacy device and return
 | 
			
		||||
			 * RESERVATION CONFLICT on some CDBs */
 | 
			
		||||
 | 
			
		||||
	se_deve = se_sess->se_node_acl->device_list[cmd->orig_fe_lun];
 | 
			
		||||
	rcu_read_lock();
 | 
			
		||||
	se_deve = target_nacl_find_deve(nacl, cmd->orig_fe_lun);
 | 
			
		||||
	/*
 | 
			
		||||
	 * Determine if the registration should be ignored due to
 | 
			
		||||
	 * non-matching ISIDs in target_scsi3_pr_reservation_check().
 | 
			
		||||
| 
						 | 
				
			
			@ -368,8 +370,10 @@ static int core_scsi3_pr_seq_non_holder(
 | 
			
		|||
			registered_nexus = 1;
 | 
			
		||||
		break;
 | 
			
		||||
	default:
 | 
			
		||||
		rcu_read_unlock();
 | 
			
		||||
		return -EINVAL;
 | 
			
		||||
	}
 | 
			
		||||
	rcu_read_unlock();
 | 
			
		||||
	/*
 | 
			
		||||
	 * Referenced from spc4r17 table 45 for *NON* PR holder access
 | 
			
		||||
	 */
 | 
			
		||||
| 
						 | 
				
			
			@ -735,7 +739,7 @@ static struct t10_pr_registration *__core_scsi3_alloc_registration(
 | 
			
		|||
			if (strcmp(nacl->initiatorname, nacl_tmp->initiatorname))
 | 
			
		||||
				continue;
 | 
			
		||||
 | 
			
		||||
			atomic_inc_mb(&deve_tmp->pr_ref_count);
 | 
			
		||||
			kref_get(&deve_tmp->pr_kref);
 | 
			
		||||
			spin_unlock_bh(&port->sep_alua_lock);
 | 
			
		||||
			/*
 | 
			
		||||
			 * Grab a configfs group dependency that is released
 | 
			
		||||
| 
						 | 
				
			
			@ -748,7 +752,7 @@ static struct t10_pr_registration *__core_scsi3_alloc_registration(
 | 
			
		|||
				pr_err("core_scsi3_lunacl_depend"
 | 
			
		||||
						"_item() failed\n");
 | 
			
		||||
				atomic_dec_mb(&port->sep_tg_pt_ref_cnt);
 | 
			
		||||
				atomic_dec_mb(&deve_tmp->pr_ref_count);
 | 
			
		||||
				kref_put(&deve_tmp->pr_kref, target_pr_kref_release);
 | 
			
		||||
				goto out;
 | 
			
		||||
			}
 | 
			
		||||
			/*
 | 
			
		||||
| 
						 | 
				
			
			@ -763,7 +767,6 @@ static struct t10_pr_registration *__core_scsi3_alloc_registration(
 | 
			
		|||
						sa_res_key, all_tg_pt, aptpl);
 | 
			
		||||
			if (!pr_reg_atp) {
 | 
			
		||||
				atomic_dec_mb(&port->sep_tg_pt_ref_cnt);
 | 
			
		||||
				atomic_dec_mb(&deve_tmp->pr_ref_count);
 | 
			
		||||
				core_scsi3_lunacl_undepend_item(deve_tmp);
 | 
			
		||||
				goto out;
 | 
			
		||||
			}
 | 
			
		||||
| 
						 | 
				
			
			@ -896,7 +899,7 @@ static int __core_scsi3_check_aptpl_registration(
 | 
			
		|||
	struct se_lun *lun,
 | 
			
		||||
	u32 target_lun,
 | 
			
		||||
	struct se_node_acl *nacl,
 | 
			
		||||
	struct se_dev_entry *deve)
 | 
			
		||||
	u32 mapped_lun)
 | 
			
		||||
{
 | 
			
		||||
	struct t10_pr_registration *pr_reg, *pr_reg_tmp;
 | 
			
		||||
	struct t10_reservation *pr_tmpl = &dev->t10_pr;
 | 
			
		||||
| 
						 | 
				
			
			@ -924,13 +927,12 @@ static int __core_scsi3_check_aptpl_registration(
 | 
			
		|||
				pr_reg_aptpl_list) {
 | 
			
		||||
 | 
			
		||||
		if (!strcmp(pr_reg->pr_iport, i_port) &&
 | 
			
		||||
		     (pr_reg->pr_res_mapped_lun == deve->mapped_lun) &&
 | 
			
		||||
		     (pr_reg->pr_res_mapped_lun == mapped_lun) &&
 | 
			
		||||
		    !(strcmp(pr_reg->pr_tport, t_port)) &&
 | 
			
		||||
		     (pr_reg->pr_reg_tpgt == tpgt) &&
 | 
			
		||||
		     (pr_reg->pr_aptpl_target_lun == target_lun)) {
 | 
			
		||||
 | 
			
		||||
			pr_reg->pr_reg_nacl = nacl;
 | 
			
		||||
			pr_reg->pr_reg_deve = deve;
 | 
			
		||||
			pr_reg->pr_reg_tg_pt_lun = lun;
 | 
			
		||||
 | 
			
		||||
			list_del(&pr_reg->pr_reg_aptpl_list);
 | 
			
		||||
| 
						 | 
				
			
			@ -968,13 +970,12 @@ int core_scsi3_check_aptpl_registration(
 | 
			
		|||
	struct se_node_acl *nacl,
 | 
			
		||||
	u32 mapped_lun)
 | 
			
		||||
{
 | 
			
		||||
	struct se_dev_entry *deve = nacl->device_list[mapped_lun];
 | 
			
		||||
 | 
			
		||||
	if (dev->dev_reservation_flags & DRF_SPC2_RESERVATIONS)
 | 
			
		||||
		return 0;
 | 
			
		||||
 | 
			
		||||
	return __core_scsi3_check_aptpl_registration(dev, tpg, lun,
 | 
			
		||||
				lun->unpacked_lun, nacl, deve);
 | 
			
		||||
						     lun->unpacked_lun, nacl,
 | 
			
		||||
						     mapped_lun);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static void __core_scsi3_dump_registration(
 | 
			
		||||
| 
						 | 
				
			
			@ -1408,27 +1409,29 @@ static int core_scsi3_lunacl_depend_item(struct se_dev_entry *se_deve)
 | 
			
		|||
 | 
			
		||||
static void core_scsi3_lunacl_undepend_item(struct se_dev_entry *se_deve)
 | 
			
		||||
{
 | 
			
		||||
	struct se_lun_acl *lun_acl = se_deve->se_lun_acl;
 | 
			
		||||
	struct se_lun_acl *lun_acl;
 | 
			
		||||
	struct se_node_acl *nacl;
 | 
			
		||||
	struct se_portal_group *tpg;
 | 
			
		||||
	/*
 | 
			
		||||
	 * For nacl->dynamic_node_acl=1
 | 
			
		||||
	 */
 | 
			
		||||
	lun_acl = se_deve->se_lun_acl;
 | 
			
		||||
	if (!lun_acl) {
 | 
			
		||||
		atomic_dec_mb(&se_deve->pr_ref_count);
 | 
			
		||||
		kref_put(&se_deve->pr_kref, target_pr_kref_release);
 | 
			
		||||
		return;
 | 
			
		||||
	}
 | 
			
		||||
	nacl = lun_acl->se_lun_nacl;
 | 
			
		||||
	tpg = nacl->se_tpg;
 | 
			
		||||
 | 
			
		||||
	target_undepend_item(&lun_acl->se_lun_group.cg_item);
 | 
			
		||||
	atomic_dec_mb(&se_deve->pr_ref_count);
 | 
			
		||||
	kref_put(&se_deve->pr_kref, target_pr_kref_release);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static sense_reason_t
 | 
			
		||||
core_scsi3_decode_spec_i_port(
 | 
			
		||||
	struct se_cmd *cmd,
 | 
			
		||||
	struct se_portal_group *tpg,
 | 
			
		||||
	struct se_dev_entry *local_se_deve,
 | 
			
		||||
	unsigned char *l_isid,
 | 
			
		||||
	u64 sa_res_key,
 | 
			
		||||
	int all_tg_pt,
 | 
			
		||||
| 
						 | 
				
			
			@ -1439,7 +1442,7 @@ core_scsi3_decode_spec_i_port(
 | 
			
		|||
	struct se_portal_group *dest_tpg = NULL, *tmp_tpg;
 | 
			
		||||
	struct se_session *se_sess = cmd->se_sess;
 | 
			
		||||
	struct se_node_acl *dest_node_acl = NULL;
 | 
			
		||||
	struct se_dev_entry *dest_se_deve = NULL, *local_se_deve;
 | 
			
		||||
	struct se_dev_entry __rcu *dest_se_deve = NULL;
 | 
			
		||||
	struct t10_pr_registration *dest_pr_reg, *local_pr_reg, *pr_reg_e;
 | 
			
		||||
	struct t10_pr_registration *pr_reg_tmp, *pr_reg_tmp_safe;
 | 
			
		||||
	LIST_HEAD(tid_dest_list);
 | 
			
		||||
| 
						 | 
				
			
			@ -1452,7 +1455,6 @@ core_scsi3_decode_spec_i_port(
 | 
			
		|||
	int dest_local_nexus;
 | 
			
		||||
	u32 dest_rtpi = 0;
 | 
			
		||||
 | 
			
		||||
	local_se_deve = se_sess->se_node_acl->device_list[cmd->orig_fe_lun];
 | 
			
		||||
	/*
 | 
			
		||||
	 * Allocate a struct pr_transport_id_holder and setup the
 | 
			
		||||
	 * local_node_acl and local_se_deve pointers and add to
 | 
			
		||||
| 
						 | 
				
			
			@ -1467,7 +1469,6 @@ core_scsi3_decode_spec_i_port(
 | 
			
		|||
	INIT_LIST_HEAD(&tidh_new->dest_list);
 | 
			
		||||
	tidh_new->dest_tpg = tpg;
 | 
			
		||||
	tidh_new->dest_node_acl = se_sess->se_node_acl;
 | 
			
		||||
	tidh_new->dest_se_deve = local_se_deve;
 | 
			
		||||
 | 
			
		||||
	local_pr_reg = __core_scsi3_alloc_registration(cmd->se_dev,
 | 
			
		||||
				se_sess->se_node_acl, local_se_deve, l_isid,
 | 
			
		||||
| 
						 | 
				
			
			@ -1476,6 +1477,7 @@ core_scsi3_decode_spec_i_port(
 | 
			
		|||
		kfree(tidh_new);
 | 
			
		||||
		return TCM_LOGICAL_UNIT_COMMUNICATION_FAILURE;
 | 
			
		||||
	}
 | 
			
		||||
	rcu_assign_pointer(tidh_new->dest_se_deve, local_se_deve);
 | 
			
		||||
	tidh_new->dest_pr_reg = local_pr_reg;
 | 
			
		||||
	/*
 | 
			
		||||
	 * The local I_T nexus does not hold any configfs dependances,
 | 
			
		||||
| 
						 | 
				
			
			@ -1635,7 +1637,7 @@ core_scsi3_decode_spec_i_port(
 | 
			
		|||
		if (core_scsi3_lunacl_depend_item(dest_se_deve)) {
 | 
			
		||||
			pr_err("core_scsi3_lunacl_depend_item()"
 | 
			
		||||
					" failed\n");
 | 
			
		||||
			atomic_dec_mb(&dest_se_deve->pr_ref_count);
 | 
			
		||||
			kref_put(&dest_se_deve->pr_kref, target_pr_kref_release);
 | 
			
		||||
			core_scsi3_nodeacl_undepend_item(dest_node_acl);
 | 
			
		||||
			core_scsi3_tpg_undepend_item(dest_tpg);
 | 
			
		||||
			ret = TCM_LOGICAL_UNIT_COMMUNICATION_FAILURE;
 | 
			
		||||
| 
						 | 
				
			
			@ -1990,6 +1992,7 @@ core_scsi3_emulate_pro_register(struct se_cmd *cmd, u64 res_key, u64 sa_res_key,
 | 
			
		|||
		bool aptpl, bool all_tg_pt, bool spec_i_pt, enum register_type register_type)
 | 
			
		||||
{
 | 
			
		||||
	struct se_session *se_sess = cmd->se_sess;
 | 
			
		||||
	struct se_node_acl *nacl = se_sess->se_node_acl;
 | 
			
		||||
	struct se_device *dev = cmd->se_dev;
 | 
			
		||||
	struct se_dev_entry *se_deve;
 | 
			
		||||
	struct se_lun *se_lun = cmd->se_lun;
 | 
			
		||||
| 
						 | 
				
			
			@ -2005,7 +2008,14 @@ core_scsi3_emulate_pro_register(struct se_cmd *cmd, u64 res_key, u64 sa_res_key,
 | 
			
		|||
		return TCM_LOGICAL_UNIT_COMMUNICATION_FAILURE;
 | 
			
		||||
	}
 | 
			
		||||
	se_tpg = se_sess->se_tpg;
 | 
			
		||||
	se_deve = se_sess->se_node_acl->device_list[cmd->orig_fe_lun];
 | 
			
		||||
 | 
			
		||||
	rcu_read_lock();
 | 
			
		||||
	se_deve = target_nacl_find_deve(nacl, cmd->orig_fe_lun);
 | 
			
		||||
	if (!se_deve) {
 | 
			
		||||
		pr_err("Unable to locate se_deve for PRO-REGISTER\n");
 | 
			
		||||
		rcu_read_unlock();
 | 
			
		||||
		return TCM_LOGICAL_UNIT_COMMUNICATION_FAILURE;
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if (se_tpg->se_tpg_tfo->sess_get_initiator_sid) {
 | 
			
		||||
		memset(&isid_buf[0], 0, PR_REG_ISID_LEN);
 | 
			
		||||
| 
						 | 
				
			
			@ -2021,14 +2031,16 @@ core_scsi3_emulate_pro_register(struct se_cmd *cmd, u64 res_key, u64 sa_res_key,
 | 
			
		|||
		if (res_key) {
 | 
			
		||||
			pr_warn("SPC-3 PR: Reservation Key non-zero"
 | 
			
		||||
				" for SA REGISTER, returning CONFLICT\n");
 | 
			
		||||
			rcu_read_unlock();
 | 
			
		||||
			return TCM_RESERVATION_CONFLICT;
 | 
			
		||||
		}
 | 
			
		||||
		/*
 | 
			
		||||
		 * Do nothing but return GOOD status.
 | 
			
		||||
		 */
 | 
			
		||||
		if (!sa_res_key)
 | 
			
		||||
		if (!sa_res_key) {
 | 
			
		||||
			rcu_read_unlock();
 | 
			
		||||
			return 0;
 | 
			
		||||
 | 
			
		||||
		}
 | 
			
		||||
		if (!spec_i_pt) {
 | 
			
		||||
			/*
 | 
			
		||||
			 * Perform the Service Action REGISTER on the Initiator
 | 
			
		||||
| 
						 | 
				
			
			@ -2041,6 +2053,7 @@ core_scsi3_emulate_pro_register(struct se_cmd *cmd, u64 res_key, u64 sa_res_key,
 | 
			
		|||
					register_type, 0)) {
 | 
			
		||||
				pr_err("Unable to allocate"
 | 
			
		||||
					" struct t10_pr_registration\n");
 | 
			
		||||
				rcu_read_unlock();
 | 
			
		||||
				return TCM_INVALID_PARAMETER_LIST;
 | 
			
		||||
			}
 | 
			
		||||
		} else {
 | 
			
		||||
| 
						 | 
				
			
			@ -2052,14 +2065,17 @@ core_scsi3_emulate_pro_register(struct se_cmd *cmd, u64 res_key, u64 sa_res_key,
 | 
			
		|||
			 * logic from of core_scsi3_alloc_registration() for
 | 
			
		||||
			 * each TransportID provided SCSI Initiator Port/Device
 | 
			
		||||
			 */
 | 
			
		||||
			ret = core_scsi3_decode_spec_i_port(cmd, se_tpg,
 | 
			
		||||
			ret = core_scsi3_decode_spec_i_port(cmd, se_tpg, se_deve,
 | 
			
		||||
					isid_ptr, sa_res_key, all_tg_pt, aptpl);
 | 
			
		||||
			if (ret != 0)
 | 
			
		||||
			if (ret != 0) {
 | 
			
		||||
				rcu_read_unlock();
 | 
			
		||||
				return ret;
 | 
			
		||||
			}
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		rcu_read_unlock();
 | 
			
		||||
		return core_scsi3_update_and_write_aptpl(dev, aptpl);
 | 
			
		||||
	}
 | 
			
		||||
	rcu_read_unlock();
 | 
			
		||||
 | 
			
		||||
	/* ok, existing registration */
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -3321,7 +3337,7 @@ core_scsi3_emulate_pro_register_and_move(struct se_cmd *cmd, u64 res_key,
 | 
			
		|||
 | 
			
		||||
	if (core_scsi3_lunacl_depend_item(dest_se_deve)) {
 | 
			
		||||
		pr_err("core_scsi3_lunacl_depend_item() failed\n");
 | 
			
		||||
		atomic_dec_mb(&dest_se_deve->pr_ref_count);
 | 
			
		||||
		kref_put(&dest_se_deve->pr_kref, target_pr_kref_release);
 | 
			
		||||
		dest_se_deve = NULL;
 | 
			
		||||
		ret = TCM_LOGICAL_UNIT_COMMUNICATION_FAILURE;
 | 
			
		||||
		goto out;
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -47,6 +47,7 @@
 | 
			
		|||
#include <target/target_core_backend_configfs.h>
 | 
			
		||||
 | 
			
		||||
#include "target_core_alua.h"
 | 
			
		||||
#include "target_core_internal.h"
 | 
			
		||||
#include "target_core_pscsi.h"
 | 
			
		||||
 | 
			
		||||
#define ISPRINT(a)  ((a >= ' ') && (a <= '~'))
 | 
			
		||||
| 
						 | 
				
			
			@ -637,12 +638,14 @@ static void pscsi_transport_complete(struct se_cmd *cmd, struct scatterlist *sg,
 | 
			
		|||
	 * Hack to make sure that Write-Protect modepage is set if R/O mode is
 | 
			
		||||
	 * forced.
 | 
			
		||||
	 */
 | 
			
		||||
	if (!cmd->se_deve || !cmd->data_length)
 | 
			
		||||
	if (!cmd->data_length)
 | 
			
		||||
		goto after_mode_sense;
 | 
			
		||||
 | 
			
		||||
	if (((cdb[0] == MODE_SENSE) || (cdb[0] == MODE_SENSE_10)) &&
 | 
			
		||||
	     (status_byte(result) << 1) == SAM_STAT_GOOD) {
 | 
			
		||||
		if (cmd->se_deve->lun_flags & TRANSPORT_LUNFLAGS_READ_ONLY) {
 | 
			
		||||
		bool read_only = target_lun_is_rdonly(cmd);
 | 
			
		||||
 | 
			
		||||
		if (read_only) {
 | 
			
		||||
			unsigned char *buf;
 | 
			
		||||
 | 
			
		||||
			buf = transport_kmap_data_sg(cmd);
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -981,6 +981,7 @@ static sense_reason_t spc_emulate_modesense(struct se_cmd *cmd)
 | 
			
		|||
	int length = 0;
 | 
			
		||||
	int ret;
 | 
			
		||||
	int i;
 | 
			
		||||
	bool read_only = target_lun_is_rdonly(cmd);;
 | 
			
		||||
 | 
			
		||||
	memset(buf, 0, SE_MODE_PAGE_BUF);
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -991,9 +992,7 @@ static sense_reason_t spc_emulate_modesense(struct se_cmd *cmd)
 | 
			
		|||
	length = ten ? 3 : 2;
 | 
			
		||||
 | 
			
		||||
	/* DEVICE-SPECIFIC PARAMETER */
 | 
			
		||||
	if ((cmd->se_lun->lun_access & TRANSPORT_LUNFLAGS_READ_ONLY) ||
 | 
			
		||||
	    (cmd->se_deve &&
 | 
			
		||||
	     (cmd->se_deve->lun_flags & TRANSPORT_LUNFLAGS_READ_ONLY)))
 | 
			
		||||
	if ((cmd->se_lun->lun_access & TRANSPORT_LUNFLAGS_READ_ONLY) || read_only)
 | 
			
		||||
		spc_modesense_write_protect(&buf[length], type);
 | 
			
		||||
 | 
			
		||||
	/*
 | 
			
		||||
| 
						 | 
				
			
			@ -1211,8 +1210,9 @@ sense_reason_t spc_emulate_report_luns(struct se_cmd *cmd)
 | 
			
		|||
{
 | 
			
		||||
	struct se_dev_entry *deve;
 | 
			
		||||
	struct se_session *sess = cmd->se_sess;
 | 
			
		||||
	struct se_node_acl *nacl;
 | 
			
		||||
	unsigned char *buf;
 | 
			
		||||
	u32 lun_count = 0, offset = 8, i;
 | 
			
		||||
	u32 lun_count = 0, offset = 8;
 | 
			
		||||
 | 
			
		||||
	if (cmd->data_length < 16) {
 | 
			
		||||
		pr_warn("REPORT LUNS allocation length %u too small\n",
 | 
			
		||||
| 
						 | 
				
			
			@ -1234,12 +1234,10 @@ sense_reason_t spc_emulate_report_luns(struct se_cmd *cmd)
 | 
			
		|||
		lun_count = 1;
 | 
			
		||||
		goto done;
 | 
			
		||||
	}
 | 
			
		||||
	nacl = sess->se_node_acl;
 | 
			
		||||
 | 
			
		||||
	spin_lock_irq(&sess->se_node_acl->device_list_lock);
 | 
			
		||||
	for (i = 0; i < TRANSPORT_MAX_LUNS_PER_TPG; i++) {
 | 
			
		||||
		deve = sess->se_node_acl->device_list[i];
 | 
			
		||||
		if (!(deve->lun_flags & TRANSPORT_LUNFLAGS_INITIATOR_ACCESS))
 | 
			
		||||
			continue;
 | 
			
		||||
	rcu_read_lock();
 | 
			
		||||
	hlist_for_each_entry_rcu(deve, &nacl->lun_entry_hlist, link) {
 | 
			
		||||
		/*
 | 
			
		||||
		 * We determine the correct LUN LIST LENGTH even once we
 | 
			
		||||
		 * have reached the initial allocation length.
 | 
			
		||||
| 
						 | 
				
			
			@ -1252,7 +1250,7 @@ sense_reason_t spc_emulate_report_luns(struct se_cmd *cmd)
 | 
			
		|||
		int_to_scsilun(deve->mapped_lun, (struct scsi_lun *)&buf[offset]);
 | 
			
		||||
		offset += 8;
 | 
			
		||||
	}
 | 
			
		||||
	spin_unlock_irq(&sess->se_node_acl->device_list_lock);
 | 
			
		||||
	rcu_read_unlock();
 | 
			
		||||
 | 
			
		||||
	/*
 | 
			
		||||
	 * See SPC3 r07, page 159.
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -1084,17 +1084,17 @@ static ssize_t target_stat_scsi_auth_intr_show_attr_inst(
 | 
			
		|||
	struct se_portal_group *tpg;
 | 
			
		||||
	ssize_t ret;
 | 
			
		||||
 | 
			
		||||
	spin_lock_irq(&nacl->device_list_lock);
 | 
			
		||||
	deve = nacl->device_list[lacl->mapped_lun];
 | 
			
		||||
	if (!deve->se_lun || !deve->se_lun_acl) {
 | 
			
		||||
		spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
	rcu_read_lock();
 | 
			
		||||
	deve = target_nacl_find_deve(nacl, lacl->mapped_lun);
 | 
			
		||||
	if (!deve) {
 | 
			
		||||
		rcu_read_unlock();
 | 
			
		||||
		return -ENODEV;
 | 
			
		||||
	}
 | 
			
		||||
	tpg = nacl->se_tpg;
 | 
			
		||||
	/* scsiInstIndex */
 | 
			
		||||
	ret = snprintf(page, PAGE_SIZE, "%u\n",
 | 
			
		||||
			tpg->se_tpg_tfo->tpg_get_inst_index(tpg));
 | 
			
		||||
	spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
	rcu_read_unlock();
 | 
			
		||||
	return ret;
 | 
			
		||||
}
 | 
			
		||||
DEV_STAT_SCSI_AUTH_INTR_ATTR_RO(inst);
 | 
			
		||||
| 
						 | 
				
			
			@ -1109,16 +1109,16 @@ static ssize_t target_stat_scsi_auth_intr_show_attr_dev(
 | 
			
		|||
	struct se_lun *lun;
 | 
			
		||||
	ssize_t ret;
 | 
			
		||||
 | 
			
		||||
	spin_lock_irq(&nacl->device_list_lock);
 | 
			
		||||
	deve = nacl->device_list[lacl->mapped_lun];
 | 
			
		||||
	if (!deve->se_lun || !deve->se_lun_acl) {
 | 
			
		||||
		spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
	rcu_read_lock();
 | 
			
		||||
	deve = target_nacl_find_deve(nacl, lacl->mapped_lun);
 | 
			
		||||
	if (!deve) {
 | 
			
		||||
		rcu_read_unlock();
 | 
			
		||||
		return -ENODEV;
 | 
			
		||||
	}
 | 
			
		||||
	lun = deve->se_lun;
 | 
			
		||||
	lun = rcu_dereference(deve->se_lun);
 | 
			
		||||
	/* scsiDeviceIndex */
 | 
			
		||||
	ret = snprintf(page, PAGE_SIZE, "%u\n", lun->lun_se_dev->dev_index);
 | 
			
		||||
	spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
	ret = snprintf(page, PAGE_SIZE, "%u\n", lun->lun_index);
 | 
			
		||||
	rcu_read_unlock();
 | 
			
		||||
	return ret;
 | 
			
		||||
}
 | 
			
		||||
DEV_STAT_SCSI_AUTH_INTR_ATTR_RO(dev);
 | 
			
		||||
| 
						 | 
				
			
			@ -1133,16 +1133,16 @@ static ssize_t target_stat_scsi_auth_intr_show_attr_port(
 | 
			
		|||
	struct se_portal_group *tpg;
 | 
			
		||||
	ssize_t ret;
 | 
			
		||||
 | 
			
		||||
	spin_lock_irq(&nacl->device_list_lock);
 | 
			
		||||
	deve = nacl->device_list[lacl->mapped_lun];
 | 
			
		||||
	if (!deve->se_lun || !deve->se_lun_acl) {
 | 
			
		||||
		spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
	rcu_read_lock();
 | 
			
		||||
	deve = target_nacl_find_deve(nacl, lacl->mapped_lun);
 | 
			
		||||
	if (!deve) {
 | 
			
		||||
		rcu_read_unlock();
 | 
			
		||||
		return -ENODEV;
 | 
			
		||||
	}
 | 
			
		||||
	tpg = nacl->se_tpg;
 | 
			
		||||
	/* scsiAuthIntrTgtPortIndex */
 | 
			
		||||
	ret = snprintf(page, PAGE_SIZE, "%u\n", tpg->se_tpg_tfo->tpg_get_tag(tpg));
 | 
			
		||||
	spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
	rcu_read_unlock();
 | 
			
		||||
	return ret;
 | 
			
		||||
}
 | 
			
		||||
DEV_STAT_SCSI_AUTH_INTR_ATTR_RO(port);
 | 
			
		||||
| 
						 | 
				
			
			@ -1156,15 +1156,15 @@ static ssize_t target_stat_scsi_auth_intr_show_attr_indx(
 | 
			
		|||
	struct se_dev_entry *deve;
 | 
			
		||||
	ssize_t ret;
 | 
			
		||||
 | 
			
		||||
	spin_lock_irq(&nacl->device_list_lock);
 | 
			
		||||
	deve = nacl->device_list[lacl->mapped_lun];
 | 
			
		||||
	if (!deve->se_lun || !deve->se_lun_acl) {
 | 
			
		||||
		spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
	rcu_read_lock();
 | 
			
		||||
	deve = target_nacl_find_deve(nacl, lacl->mapped_lun);
 | 
			
		||||
	if (!deve) {
 | 
			
		||||
		rcu_read_unlock();
 | 
			
		||||
		return -ENODEV;
 | 
			
		||||
	}
 | 
			
		||||
	/* scsiAuthIntrIndex */
 | 
			
		||||
	ret = snprintf(page, PAGE_SIZE, "%u\n", nacl->acl_index);
 | 
			
		||||
	spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
	rcu_read_unlock();
 | 
			
		||||
	return ret;
 | 
			
		||||
}
 | 
			
		||||
DEV_STAT_SCSI_AUTH_INTR_ATTR_RO(indx);
 | 
			
		||||
| 
						 | 
				
			
			@ -1178,15 +1178,15 @@ static ssize_t target_stat_scsi_auth_intr_show_attr_dev_or_port(
 | 
			
		|||
	struct se_dev_entry *deve;
 | 
			
		||||
	ssize_t ret;
 | 
			
		||||
 | 
			
		||||
	spin_lock_irq(&nacl->device_list_lock);
 | 
			
		||||
	deve = nacl->device_list[lacl->mapped_lun];
 | 
			
		||||
	if (!deve->se_lun || !deve->se_lun_acl) {
 | 
			
		||||
		spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
	rcu_read_lock();
 | 
			
		||||
	deve = target_nacl_find_deve(nacl, lacl->mapped_lun);
 | 
			
		||||
	if (!deve) {
 | 
			
		||||
		rcu_read_unlock();
 | 
			
		||||
		return -ENODEV;
 | 
			
		||||
	}
 | 
			
		||||
	/* scsiAuthIntrDevOrPort */
 | 
			
		||||
	ret = snprintf(page, PAGE_SIZE, "%u\n", 1);
 | 
			
		||||
	spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
	rcu_read_unlock();
 | 
			
		||||
	return ret;
 | 
			
		||||
}
 | 
			
		||||
DEV_STAT_SCSI_AUTH_INTR_ATTR_RO(dev_or_port);
 | 
			
		||||
| 
						 | 
				
			
			@ -1200,15 +1200,15 @@ static ssize_t target_stat_scsi_auth_intr_show_attr_intr_name(
 | 
			
		|||
	struct se_dev_entry *deve;
 | 
			
		||||
	ssize_t ret;
 | 
			
		||||
 | 
			
		||||
	spin_lock_irq(&nacl->device_list_lock);
 | 
			
		||||
	deve = nacl->device_list[lacl->mapped_lun];
 | 
			
		||||
	if (!deve->se_lun || !deve->se_lun_acl) {
 | 
			
		||||
		spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
	rcu_read_lock();
 | 
			
		||||
	deve = target_nacl_find_deve(nacl, lacl->mapped_lun);
 | 
			
		||||
	if (!deve) {
 | 
			
		||||
		rcu_read_unlock();
 | 
			
		||||
		return -ENODEV;
 | 
			
		||||
	}
 | 
			
		||||
	/* scsiAuthIntrName */
 | 
			
		||||
	ret = snprintf(page, PAGE_SIZE, "%s\n", nacl->initiatorname);
 | 
			
		||||
	spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
	rcu_read_unlock();
 | 
			
		||||
	return ret;
 | 
			
		||||
}
 | 
			
		||||
DEV_STAT_SCSI_AUTH_INTR_ATTR_RO(intr_name);
 | 
			
		||||
| 
						 | 
				
			
			@ -1222,15 +1222,15 @@ static ssize_t target_stat_scsi_auth_intr_show_attr_map_indx(
 | 
			
		|||
	struct se_dev_entry *deve;
 | 
			
		||||
	ssize_t ret;
 | 
			
		||||
 | 
			
		||||
	spin_lock_irq(&nacl->device_list_lock);
 | 
			
		||||
	deve = nacl->device_list[lacl->mapped_lun];
 | 
			
		||||
	if (!deve->se_lun || !deve->se_lun_acl) {
 | 
			
		||||
		spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
	rcu_read_lock();
 | 
			
		||||
	deve = target_nacl_find_deve(nacl, lacl->mapped_lun);
 | 
			
		||||
	if (!deve) {
 | 
			
		||||
		rcu_read_unlock();
 | 
			
		||||
		return -ENODEV;
 | 
			
		||||
	}
 | 
			
		||||
	/* FIXME: scsiAuthIntrLunMapIndex */
 | 
			
		||||
	ret = snprintf(page, PAGE_SIZE, "%u\n", 0);
 | 
			
		||||
	spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
	rcu_read_unlock();
 | 
			
		||||
	return ret;
 | 
			
		||||
}
 | 
			
		||||
DEV_STAT_SCSI_AUTH_INTR_ATTR_RO(map_indx);
 | 
			
		||||
| 
						 | 
				
			
			@ -1244,15 +1244,15 @@ static ssize_t target_stat_scsi_auth_intr_show_attr_att_count(
 | 
			
		|||
	struct se_dev_entry *deve;
 | 
			
		||||
	ssize_t ret;
 | 
			
		||||
 | 
			
		||||
	spin_lock_irq(&nacl->device_list_lock);
 | 
			
		||||
	deve = nacl->device_list[lacl->mapped_lun];
 | 
			
		||||
	if (!deve->se_lun || !deve->se_lun_acl) {
 | 
			
		||||
		spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
	rcu_read_lock();
 | 
			
		||||
	deve = target_nacl_find_deve(nacl, lacl->mapped_lun);
 | 
			
		||||
	if (!deve) {
 | 
			
		||||
		rcu_read_unlock();
 | 
			
		||||
		return -ENODEV;
 | 
			
		||||
	}
 | 
			
		||||
	/* scsiAuthIntrAttachedTimes */
 | 
			
		||||
	ret = snprintf(page, PAGE_SIZE, "%u\n", deve->attach_count);
 | 
			
		||||
	spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
	rcu_read_unlock();
 | 
			
		||||
	return ret;
 | 
			
		||||
}
 | 
			
		||||
DEV_STAT_SCSI_AUTH_INTR_ATTR_RO(att_count);
 | 
			
		||||
| 
						 | 
				
			
			@ -1266,15 +1266,16 @@ static ssize_t target_stat_scsi_auth_intr_show_attr_num_cmds(
 | 
			
		|||
	struct se_dev_entry *deve;
 | 
			
		||||
	ssize_t ret;
 | 
			
		||||
 | 
			
		||||
	spin_lock_irq(&nacl->device_list_lock);
 | 
			
		||||
	deve = nacl->device_list[lacl->mapped_lun];
 | 
			
		||||
	if (!deve->se_lun || !deve->se_lun_acl) {
 | 
			
		||||
		spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
	rcu_read_lock();
 | 
			
		||||
	deve = target_nacl_find_deve(nacl, lacl->mapped_lun);
 | 
			
		||||
	if (!deve) {
 | 
			
		||||
		rcu_read_unlock();
 | 
			
		||||
		return -ENODEV;
 | 
			
		||||
	}
 | 
			
		||||
	/* scsiAuthIntrOutCommands */
 | 
			
		||||
	ret = snprintf(page, PAGE_SIZE, "%u\n", deve->total_cmds);
 | 
			
		||||
	spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
	ret = snprintf(page, PAGE_SIZE, "%lu\n",
 | 
			
		||||
		       atomic_long_read(&deve->total_cmds));
 | 
			
		||||
	rcu_read_unlock();
 | 
			
		||||
	return ret;
 | 
			
		||||
}
 | 
			
		||||
DEV_STAT_SCSI_AUTH_INTR_ATTR_RO(num_cmds);
 | 
			
		||||
| 
						 | 
				
			
			@ -1288,15 +1289,16 @@ static ssize_t target_stat_scsi_auth_intr_show_attr_read_mbytes(
 | 
			
		|||
	struct se_dev_entry *deve;
 | 
			
		||||
	ssize_t ret;
 | 
			
		||||
 | 
			
		||||
	spin_lock_irq(&nacl->device_list_lock);
 | 
			
		||||
	deve = nacl->device_list[lacl->mapped_lun];
 | 
			
		||||
	if (!deve->se_lun || !deve->se_lun_acl) {
 | 
			
		||||
		spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
	rcu_read_lock();
 | 
			
		||||
	deve = target_nacl_find_deve(nacl, lacl->mapped_lun);
 | 
			
		||||
	if (!deve) {
 | 
			
		||||
		rcu_read_unlock();
 | 
			
		||||
		return -ENODEV;
 | 
			
		||||
	}
 | 
			
		||||
	/* scsiAuthIntrReadMegaBytes */
 | 
			
		||||
	ret = snprintf(page, PAGE_SIZE, "%u\n", (u32)(deve->read_bytes >> 20));
 | 
			
		||||
	spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
	ret = snprintf(page, PAGE_SIZE, "%u\n",
 | 
			
		||||
		      (u32)(atomic_long_read(&deve->read_bytes) >> 20));
 | 
			
		||||
	rcu_read_unlock();
 | 
			
		||||
	return ret;
 | 
			
		||||
}
 | 
			
		||||
DEV_STAT_SCSI_AUTH_INTR_ATTR_RO(read_mbytes);
 | 
			
		||||
| 
						 | 
				
			
			@ -1310,15 +1312,16 @@ static ssize_t target_stat_scsi_auth_intr_show_attr_write_mbytes(
 | 
			
		|||
	struct se_dev_entry *deve;
 | 
			
		||||
	ssize_t ret;
 | 
			
		||||
 | 
			
		||||
	spin_lock_irq(&nacl->device_list_lock);
 | 
			
		||||
	deve = nacl->device_list[lacl->mapped_lun];
 | 
			
		||||
	if (!deve->se_lun || !deve->se_lun_acl) {
 | 
			
		||||
		spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
	rcu_read_lock();
 | 
			
		||||
	deve = target_nacl_find_deve(nacl, lacl->mapped_lun);
 | 
			
		||||
	if (!deve) {
 | 
			
		||||
		rcu_read_unlock();
 | 
			
		||||
		return -ENODEV;
 | 
			
		||||
	}
 | 
			
		||||
	/* scsiAuthIntrWrittenMegaBytes */
 | 
			
		||||
	ret = snprintf(page, PAGE_SIZE, "%u\n", (u32)(deve->write_bytes >> 20));
 | 
			
		||||
	spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
	ret = snprintf(page, PAGE_SIZE, "%u\n",
 | 
			
		||||
		      (u32)(atomic_long_read(&deve->write_bytes) >> 20));
 | 
			
		||||
	rcu_read_unlock();
 | 
			
		||||
	return ret;
 | 
			
		||||
}
 | 
			
		||||
DEV_STAT_SCSI_AUTH_INTR_ATTR_RO(write_mbytes);
 | 
			
		||||
| 
						 | 
				
			
			@ -1332,15 +1335,15 @@ static ssize_t target_stat_scsi_auth_intr_show_attr_hs_num_cmds(
 | 
			
		|||
	struct se_dev_entry *deve;
 | 
			
		||||
	ssize_t ret;
 | 
			
		||||
 | 
			
		||||
	spin_lock_irq(&nacl->device_list_lock);
 | 
			
		||||
	deve = nacl->device_list[lacl->mapped_lun];
 | 
			
		||||
	if (!deve->se_lun || !deve->se_lun_acl) {
 | 
			
		||||
		spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
	rcu_read_lock();
 | 
			
		||||
	deve = target_nacl_find_deve(nacl, lacl->mapped_lun);
 | 
			
		||||
	if (!deve) {
 | 
			
		||||
		rcu_read_unlock();
 | 
			
		||||
		return -ENODEV;
 | 
			
		||||
	}
 | 
			
		||||
	/* FIXME: scsiAuthIntrHSOutCommands */
 | 
			
		||||
	ret = snprintf(page, PAGE_SIZE, "%u\n", 0);
 | 
			
		||||
	spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
	rcu_read_unlock();
 | 
			
		||||
	return ret;
 | 
			
		||||
}
 | 
			
		||||
DEV_STAT_SCSI_AUTH_INTR_ATTR_RO(hs_num_cmds);
 | 
			
		||||
| 
						 | 
				
			
			@ -1354,16 +1357,16 @@ static ssize_t target_stat_scsi_auth_intr_show_attr_creation_time(
 | 
			
		|||
	struct se_dev_entry *deve;
 | 
			
		||||
	ssize_t ret;
 | 
			
		||||
 | 
			
		||||
	spin_lock_irq(&nacl->device_list_lock);
 | 
			
		||||
	deve = nacl->device_list[lacl->mapped_lun];
 | 
			
		||||
	if (!deve->se_lun || !deve->se_lun_acl) {
 | 
			
		||||
		spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
	rcu_read_lock();
 | 
			
		||||
	deve = target_nacl_find_deve(nacl, lacl->mapped_lun);
 | 
			
		||||
	if (!deve) {
 | 
			
		||||
		rcu_read_unlock();
 | 
			
		||||
		return -ENODEV;
 | 
			
		||||
	}
 | 
			
		||||
	/* scsiAuthIntrLastCreation */
 | 
			
		||||
	ret = snprintf(page, PAGE_SIZE, "%u\n", (u32)(((u32)deve->creation_time -
 | 
			
		||||
				INITIAL_JIFFIES) * 100 / HZ));
 | 
			
		||||
	spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
	rcu_read_unlock();
 | 
			
		||||
	return ret;
 | 
			
		||||
}
 | 
			
		||||
DEV_STAT_SCSI_AUTH_INTR_ATTR_RO(creation_time);
 | 
			
		||||
| 
						 | 
				
			
			@ -1377,15 +1380,15 @@ static ssize_t target_stat_scsi_auth_intr_show_attr_row_status(
 | 
			
		|||
	struct se_dev_entry *deve;
 | 
			
		||||
	ssize_t ret;
 | 
			
		||||
 | 
			
		||||
	spin_lock_irq(&nacl->device_list_lock);
 | 
			
		||||
	deve = nacl->device_list[lacl->mapped_lun];
 | 
			
		||||
	if (!deve->se_lun || !deve->se_lun_acl) {
 | 
			
		||||
		spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
	rcu_read_lock();
 | 
			
		||||
	deve = target_nacl_find_deve(nacl, lacl->mapped_lun);
 | 
			
		||||
	if (!deve) {
 | 
			
		||||
		rcu_read_unlock();
 | 
			
		||||
		return -ENODEV;
 | 
			
		||||
	}
 | 
			
		||||
	/* FIXME: scsiAuthIntrRowStatus */
 | 
			
		||||
	ret = snprintf(page, PAGE_SIZE, "Ready\n");
 | 
			
		||||
	spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
	rcu_read_unlock();
 | 
			
		||||
	return ret;
 | 
			
		||||
}
 | 
			
		||||
DEV_STAT_SCSI_AUTH_INTR_ATTR_RO(row_status);
 | 
			
		||||
| 
						 | 
				
			
			@ -1450,17 +1453,17 @@ static ssize_t target_stat_scsi_att_intr_port_show_attr_inst(
 | 
			
		|||
	struct se_portal_group *tpg;
 | 
			
		||||
	ssize_t ret;
 | 
			
		||||
 | 
			
		||||
	spin_lock_irq(&nacl->device_list_lock);
 | 
			
		||||
	deve = nacl->device_list[lacl->mapped_lun];
 | 
			
		||||
	if (!deve->se_lun || !deve->se_lun_acl) {
 | 
			
		||||
		spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
	rcu_read_lock();
 | 
			
		||||
	deve = target_nacl_find_deve(nacl, lacl->mapped_lun);
 | 
			
		||||
	if (!deve) {
 | 
			
		||||
		rcu_read_unlock();
 | 
			
		||||
		return -ENODEV;
 | 
			
		||||
	}
 | 
			
		||||
	tpg = nacl->se_tpg;
 | 
			
		||||
	/* scsiInstIndex */
 | 
			
		||||
	ret = snprintf(page, PAGE_SIZE, "%u\n",
 | 
			
		||||
			tpg->se_tpg_tfo->tpg_get_inst_index(tpg));
 | 
			
		||||
	spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
	rcu_read_unlock();
 | 
			
		||||
	return ret;
 | 
			
		||||
}
 | 
			
		||||
DEV_STAT_SCSI_ATTR_INTR_PORT_ATTR_RO(inst);
 | 
			
		||||
| 
						 | 
				
			
			@ -1475,16 +1478,16 @@ static ssize_t target_stat_scsi_att_intr_port_show_attr_dev(
 | 
			
		|||
	struct se_lun *lun;
 | 
			
		||||
	ssize_t ret;
 | 
			
		||||
 | 
			
		||||
	spin_lock_irq(&nacl->device_list_lock);
 | 
			
		||||
	deve = nacl->device_list[lacl->mapped_lun];
 | 
			
		||||
	if (!deve->se_lun || !deve->se_lun_acl) {
 | 
			
		||||
		spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
	rcu_read_lock();
 | 
			
		||||
	deve = target_nacl_find_deve(nacl, lacl->mapped_lun);
 | 
			
		||||
	if (!deve) {
 | 
			
		||||
		rcu_read_unlock();
 | 
			
		||||
		return -ENODEV;
 | 
			
		||||
	}
 | 
			
		||||
	lun = deve->se_lun;
 | 
			
		||||
	lun = rcu_dereference(deve->se_lun);
 | 
			
		||||
	/* scsiDeviceIndex */
 | 
			
		||||
	ret = snprintf(page, PAGE_SIZE, "%u\n", lun->lun_se_dev->dev_index);
 | 
			
		||||
	spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
	ret = snprintf(page, PAGE_SIZE, "%u\n", lun->lun_index);
 | 
			
		||||
	rcu_read_unlock();
 | 
			
		||||
	return ret;
 | 
			
		||||
}
 | 
			
		||||
DEV_STAT_SCSI_ATTR_INTR_PORT_ATTR_RO(dev);
 | 
			
		||||
| 
						 | 
				
			
			@ -1499,16 +1502,16 @@ static ssize_t target_stat_scsi_att_intr_port_show_attr_port(
 | 
			
		|||
	struct se_portal_group *tpg;
 | 
			
		||||
	ssize_t ret;
 | 
			
		||||
 | 
			
		||||
	spin_lock_irq(&nacl->device_list_lock);
 | 
			
		||||
	deve = nacl->device_list[lacl->mapped_lun];
 | 
			
		||||
	if (!deve->se_lun || !deve->se_lun_acl) {
 | 
			
		||||
		spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
	rcu_read_lock();
 | 
			
		||||
	deve = target_nacl_find_deve(nacl, lacl->mapped_lun);
 | 
			
		||||
	if (!deve) {
 | 
			
		||||
		rcu_read_unlock();
 | 
			
		||||
		return -ENODEV;
 | 
			
		||||
	}
 | 
			
		||||
	tpg = nacl->se_tpg;
 | 
			
		||||
	/* scsiPortIndex */
 | 
			
		||||
	ret = snprintf(page, PAGE_SIZE, "%u\n", tpg->se_tpg_tfo->tpg_get_tag(tpg));
 | 
			
		||||
	spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
	rcu_read_unlock();
 | 
			
		||||
	return ret;
 | 
			
		||||
}
 | 
			
		||||
DEV_STAT_SCSI_ATTR_INTR_PORT_ATTR_RO(port);
 | 
			
		||||
| 
						 | 
				
			
			@ -1548,15 +1551,15 @@ static ssize_t target_stat_scsi_att_intr_port_show_attr_port_auth_indx(
 | 
			
		|||
	struct se_dev_entry *deve;
 | 
			
		||||
	ssize_t ret;
 | 
			
		||||
 | 
			
		||||
	spin_lock_irq(&nacl->device_list_lock);
 | 
			
		||||
	deve = nacl->device_list[lacl->mapped_lun];
 | 
			
		||||
	if (!deve->se_lun || !deve->se_lun_acl) {
 | 
			
		||||
		spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
	rcu_read_lock();
 | 
			
		||||
	deve = target_nacl_find_deve(nacl, lacl->mapped_lun);
 | 
			
		||||
	if (!deve) {
 | 
			
		||||
		rcu_read_unlock();
 | 
			
		||||
		return -ENODEV;
 | 
			
		||||
	}
 | 
			
		||||
	/* scsiAttIntrPortAuthIntrIdx */
 | 
			
		||||
	ret = snprintf(page, PAGE_SIZE, "%u\n", nacl->acl_index);
 | 
			
		||||
	spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
	rcu_read_unlock();
 | 
			
		||||
	return ret;
 | 
			
		||||
}
 | 
			
		||||
DEV_STAT_SCSI_ATTR_INTR_PORT_ATTR_RO(port_auth_indx);
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -47,42 +47,6 @@ extern struct se_device *g_lun0_dev;
 | 
			
		|||
static DEFINE_SPINLOCK(tpg_lock);
 | 
			
		||||
static LIST_HEAD(tpg_list);
 | 
			
		||||
 | 
			
		||||
/*	core_clear_initiator_node_from_tpg():
 | 
			
		||||
 *
 | 
			
		||||
 *
 | 
			
		||||
 */
 | 
			
		||||
static void core_clear_initiator_node_from_tpg(
 | 
			
		||||
	struct se_node_acl *nacl,
 | 
			
		||||
	struct se_portal_group *tpg)
 | 
			
		||||
{
 | 
			
		||||
	int i;
 | 
			
		||||
	struct se_dev_entry *deve;
 | 
			
		||||
	struct se_lun *lun;
 | 
			
		||||
 | 
			
		||||
	spin_lock_irq(&nacl->device_list_lock);
 | 
			
		||||
	for (i = 0; i < TRANSPORT_MAX_LUNS_PER_TPG; i++) {
 | 
			
		||||
		deve = nacl->device_list[i];
 | 
			
		||||
 | 
			
		||||
		if (!(deve->lun_flags & TRANSPORT_LUNFLAGS_INITIATOR_ACCESS))
 | 
			
		||||
			continue;
 | 
			
		||||
 | 
			
		||||
		if (!deve->se_lun) {
 | 
			
		||||
			pr_err("%s device entries device pointer is"
 | 
			
		||||
				" NULL, but Initiator has access.\n",
 | 
			
		||||
				tpg->se_tpg_tfo->get_fabric_name());
 | 
			
		||||
			continue;
 | 
			
		||||
		}
 | 
			
		||||
 | 
			
		||||
		lun = deve->se_lun;
 | 
			
		||||
		spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
		core_disable_device_list_for_node(lun, NULL, deve->mapped_lun,
 | 
			
		||||
			TRANSPORT_LUNFLAGS_NO_ACCESS, nacl, tpg);
 | 
			
		||||
 | 
			
		||||
		spin_lock_irq(&nacl->device_list_lock);
 | 
			
		||||
	}
 | 
			
		||||
	spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/*	__core_tpg_get_initiator_node_acl():
 | 
			
		||||
 *
 | 
			
		||||
 *	spin_lock_bh(&tpg->acl_node_lock); must be held when calling
 | 
			
		||||
| 
						 | 
				
			
			@ -225,35 +189,6 @@ static void *array_zalloc(int n, size_t size, gfp_t flags)
 | 
			
		|||
	return a;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/*      core_create_device_list_for_node():
 | 
			
		||||
 *
 | 
			
		||||
 *
 | 
			
		||||
 */
 | 
			
		||||
static int core_create_device_list_for_node(struct se_node_acl *nacl)
 | 
			
		||||
{
 | 
			
		||||
	struct se_dev_entry *deve;
 | 
			
		||||
	int i;
 | 
			
		||||
 | 
			
		||||
	nacl->device_list = array_zalloc(TRANSPORT_MAX_LUNS_PER_TPG,
 | 
			
		||||
			sizeof(struct se_dev_entry), GFP_KERNEL);
 | 
			
		||||
	if (!nacl->device_list) {
 | 
			
		||||
		pr_err("Unable to allocate memory for"
 | 
			
		||||
			" struct se_node_acl->device_list\n");
 | 
			
		||||
		return -ENOMEM;
 | 
			
		||||
	}
 | 
			
		||||
	for (i = 0; i < TRANSPORT_MAX_LUNS_PER_TPG; i++) {
 | 
			
		||||
		deve = nacl->device_list[i];
 | 
			
		||||
 | 
			
		||||
		atomic_set(&deve->ua_count, 0);
 | 
			
		||||
		atomic_set(&deve->pr_ref_count, 0);
 | 
			
		||||
		spin_lock_init(&deve->ua_lock);
 | 
			
		||||
		INIT_LIST_HEAD(&deve->alua_port_list);
 | 
			
		||||
		INIT_LIST_HEAD(&deve->ua_list);
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static struct se_node_acl *target_alloc_node_acl(struct se_portal_group *tpg,
 | 
			
		||||
		const unsigned char *initiatorname)
 | 
			
		||||
{
 | 
			
		||||
| 
						 | 
				
			
			@ -266,10 +201,11 @@ static struct se_node_acl *target_alloc_node_acl(struct se_portal_group *tpg,
 | 
			
		|||
 | 
			
		||||
	INIT_LIST_HEAD(&acl->acl_list);
 | 
			
		||||
	INIT_LIST_HEAD(&acl->acl_sess_list);
 | 
			
		||||
	INIT_HLIST_HEAD(&acl->lun_entry_hlist);
 | 
			
		||||
	kref_init(&acl->acl_kref);
 | 
			
		||||
	init_completion(&acl->acl_free_comp);
 | 
			
		||||
	spin_lock_init(&acl->device_list_lock);
 | 
			
		||||
	spin_lock_init(&acl->nacl_sess_lock);
 | 
			
		||||
	mutex_init(&acl->lun_entry_mutex);
 | 
			
		||||
	atomic_set(&acl->acl_pr_ref_count, 0);
 | 
			
		||||
	if (tpg->se_tpg_tfo->tpg_get_default_depth)
 | 
			
		||||
		acl->queue_depth = tpg->se_tpg_tfo->tpg_get_default_depth(tpg);
 | 
			
		||||
| 
						 | 
				
			
			@ -281,15 +217,11 @@ static struct se_node_acl *target_alloc_node_acl(struct se_portal_group *tpg,
 | 
			
		|||
 | 
			
		||||
	tpg->se_tpg_tfo->set_default_node_attributes(acl);
 | 
			
		||||
 | 
			
		||||
	if (core_create_device_list_for_node(acl) < 0)
 | 
			
		||||
		goto out_free_acl;
 | 
			
		||||
	if (core_set_queue_depth_for_node(tpg, acl) < 0)
 | 
			
		||||
		goto out_free_device_list;
 | 
			
		||||
		goto out_free_acl;
 | 
			
		||||
 | 
			
		||||
	return acl;
 | 
			
		||||
 | 
			
		||||
out_free_device_list:
 | 
			
		||||
	core_free_device_list_for_node(acl, tpg);
 | 
			
		||||
out_free_acl:
 | 
			
		||||
	kfree(acl);
 | 
			
		||||
	return NULL;
 | 
			
		||||
| 
						 | 
				
			
			@ -454,7 +386,6 @@ void core_tpg_del_initiator_node_acl(struct se_node_acl *acl)
 | 
			
		|||
	wait_for_completion(&acl->acl_free_comp);
 | 
			
		||||
 | 
			
		||||
	core_tpg_wait_for_nacl_pr_ref(acl);
 | 
			
		||||
	core_clear_initiator_node_from_tpg(acl, tpg);
 | 
			
		||||
	core_free_device_list_for_node(acl, tpg);
 | 
			
		||||
 | 
			
		||||
	pr_debug("%s_TPG[%hu] - Deleted ACL with TCQ Depth: %d for %s"
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -50,9 +50,17 @@ target_scsi3_ua_check(struct se_cmd *cmd)
 | 
			
		|||
	if (!nacl)
 | 
			
		||||
		return 0;
 | 
			
		||||
 | 
			
		||||
	deve = nacl->device_list[cmd->orig_fe_lun];
 | 
			
		||||
	if (!atomic_read(&deve->ua_count))
 | 
			
		||||
	rcu_read_lock();
 | 
			
		||||
	deve = target_nacl_find_deve(nacl, cmd->orig_fe_lun);
 | 
			
		||||
	if (!deve) {
 | 
			
		||||
		rcu_read_unlock();
 | 
			
		||||
		return 0;
 | 
			
		||||
	}
 | 
			
		||||
	if (!atomic_read(&deve->ua_count)) {
 | 
			
		||||
		rcu_read_unlock();
 | 
			
		||||
		return 0;
 | 
			
		||||
	}
 | 
			
		||||
	rcu_read_unlock();
 | 
			
		||||
	/*
 | 
			
		||||
	 * From sam4r14, section 5.14 Unit attention condition:
 | 
			
		||||
	 *
 | 
			
		||||
| 
						 | 
				
			
			@ -103,9 +111,12 @@ int core_scsi3_ua_allocate(
 | 
			
		|||
	ua->ua_asc = asc;
 | 
			
		||||
	ua->ua_ascq = ascq;
 | 
			
		||||
 | 
			
		||||
	spin_lock_irq(&nacl->device_list_lock);
 | 
			
		||||
	deve = nacl->device_list[unpacked_lun];
 | 
			
		||||
 | 
			
		||||
	rcu_read_lock();
 | 
			
		||||
	deve = target_nacl_find_deve(nacl, unpacked_lun);
 | 
			
		||||
	if (!deve) {
 | 
			
		||||
		rcu_read_unlock();
 | 
			
		||||
		return -EINVAL;
 | 
			
		||||
	}
 | 
			
		||||
	spin_lock(&deve->ua_lock);
 | 
			
		||||
	list_for_each_entry_safe(ua_p, ua_tmp, &deve->ua_list, ua_nacl_list) {
 | 
			
		||||
		/*
 | 
			
		||||
| 
						 | 
				
			
			@ -113,7 +124,7 @@ int core_scsi3_ua_allocate(
 | 
			
		|||
		 */
 | 
			
		||||
		if ((ua_p->ua_asc == asc) && (ua_p->ua_ascq == ascq)) {
 | 
			
		||||
			spin_unlock(&deve->ua_lock);
 | 
			
		||||
			spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
			rcu_read_unlock();
 | 
			
		||||
			kmem_cache_free(se_ua_cache, ua);
 | 
			
		||||
			return 0;
 | 
			
		||||
		}
 | 
			
		||||
| 
						 | 
				
			
			@ -158,14 +169,13 @@ int core_scsi3_ua_allocate(
 | 
			
		|||
			list_add_tail(&ua->ua_nacl_list,
 | 
			
		||||
				&deve->ua_list);
 | 
			
		||||
		spin_unlock(&deve->ua_lock);
 | 
			
		||||
		spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
 | 
			
		||||
		atomic_inc_mb(&deve->ua_count);
 | 
			
		||||
		rcu_read_unlock();
 | 
			
		||||
		return 0;
 | 
			
		||||
	}
 | 
			
		||||
	list_add_tail(&ua->ua_nacl_list, &deve->ua_list);
 | 
			
		||||
	spin_unlock(&deve->ua_lock);
 | 
			
		||||
	spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
 | 
			
		||||
	pr_debug("[%s]: Allocated UNIT ATTENTION, mapped LUN: %u, ASC:"
 | 
			
		||||
		" 0x%02x, ASCQ: 0x%02x\n",
 | 
			
		||||
| 
						 | 
				
			
			@ -173,6 +183,7 @@ int core_scsi3_ua_allocate(
 | 
			
		|||
		asc, ascq);
 | 
			
		||||
 | 
			
		||||
	atomic_inc_mb(&deve->ua_count);
 | 
			
		||||
	rcu_read_unlock();
 | 
			
		||||
	return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -210,10 +221,14 @@ void core_scsi3_ua_for_check_condition(
 | 
			
		|||
	if (!nacl)
 | 
			
		||||
		return;
 | 
			
		||||
 | 
			
		||||
	spin_lock_irq(&nacl->device_list_lock);
 | 
			
		||||
	deve = nacl->device_list[cmd->orig_fe_lun];
 | 
			
		||||
	rcu_read_lock();
 | 
			
		||||
	deve = target_nacl_find_deve(nacl, cmd->orig_fe_lun);
 | 
			
		||||
	if (!deve) {
 | 
			
		||||
		rcu_read_unlock();
 | 
			
		||||
		return;
 | 
			
		||||
	}
 | 
			
		||||
	if (!atomic_read(&deve->ua_count)) {
 | 
			
		||||
		spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
		rcu_read_unlock();
 | 
			
		||||
		return;
 | 
			
		||||
	}
 | 
			
		||||
	/*
 | 
			
		||||
| 
						 | 
				
			
			@ -249,7 +264,7 @@ void core_scsi3_ua_for_check_condition(
 | 
			
		|||
		atomic_dec_mb(&deve->ua_count);
 | 
			
		||||
	}
 | 
			
		||||
	spin_unlock(&deve->ua_lock);
 | 
			
		||||
	spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
	rcu_read_unlock();
 | 
			
		||||
 | 
			
		||||
	pr_debug("[%s]: %s UNIT ATTENTION condition with"
 | 
			
		||||
		" INTLCK_CTRL: %d, mapped LUN: %u, got CDB: 0x%02x"
 | 
			
		||||
| 
						 | 
				
			
			@ -278,10 +293,14 @@ int core_scsi3_ua_clear_for_request_sense(
 | 
			
		|||
	if (!nacl)
 | 
			
		||||
		return -EINVAL;
 | 
			
		||||
 | 
			
		||||
	spin_lock_irq(&nacl->device_list_lock);
 | 
			
		||||
	deve = nacl->device_list[cmd->orig_fe_lun];
 | 
			
		||||
	rcu_read_lock();
 | 
			
		||||
	deve = target_nacl_find_deve(nacl, cmd->orig_fe_lun);
 | 
			
		||||
	if (!deve) {
 | 
			
		||||
		rcu_read_unlock();
 | 
			
		||||
		return -EINVAL;
 | 
			
		||||
	}
 | 
			
		||||
	if (!atomic_read(&deve->ua_count)) {
 | 
			
		||||
		spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
		rcu_read_unlock();
 | 
			
		||||
		return -EPERM;
 | 
			
		||||
	}
 | 
			
		||||
	/*
 | 
			
		||||
| 
						 | 
				
			
			@ -307,7 +326,7 @@ int core_scsi3_ua_clear_for_request_sense(
 | 
			
		|||
		atomic_dec_mb(&deve->ua_count);
 | 
			
		||||
	}
 | 
			
		||||
	spin_unlock(&deve->ua_lock);
 | 
			
		||||
	spin_unlock_irq(&nacl->device_list_lock);
 | 
			
		||||
	rcu_read_unlock();
 | 
			
		||||
 | 
			
		||||
	pr_debug("[%s]: Released UNIT ATTENTION condition, mapped"
 | 
			
		||||
		" LUN: %u, got REQUEST_SENSE reported ASC: 0x%02x,"
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -101,7 +101,7 @@ int	target_alloc_sgl(struct scatterlist **, unsigned int *, u32, bool);
 | 
			
		|||
sense_reason_t	transport_generic_map_mem_to_cmd(struct se_cmd *,
 | 
			
		||||
		struct scatterlist *, u32, struct scatterlist *, u32);
 | 
			
		||||
 | 
			
		||||
void	array_free(void *array, int n);
 | 
			
		||||
bool	target_lun_is_rdonly(struct se_cmd *);
 | 
			
		||||
 | 
			
		||||
/* From target_core_configfs.c to setup default backend config_item_types */
 | 
			
		||||
void	target_core_setup_sub_cits(struct se_subsystem_api *);
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -160,10 +160,8 @@ enum se_cmd_flags_table {
 | 
			
		|||
 | 
			
		||||
/* struct se_dev_entry->lun_flags and struct se_lun->lun_access */
 | 
			
		||||
enum transport_lunflags_table {
 | 
			
		||||
	TRANSPORT_LUNFLAGS_NO_ACCESS		= 0x00,
 | 
			
		||||
	TRANSPORT_LUNFLAGS_INITIATOR_ACCESS	= 0x01,
 | 
			
		||||
	TRANSPORT_LUNFLAGS_READ_ONLY		= 0x02,
 | 
			
		||||
	TRANSPORT_LUNFLAGS_READ_WRITE		= 0x04,
 | 
			
		||||
	TRANSPORT_LUNFLAGS_READ_ONLY		= 0x01,
 | 
			
		||||
	TRANSPORT_LUNFLAGS_READ_WRITE		= 0x02,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
| 
						 | 
				
			
			@ -584,10 +582,10 @@ struct se_node_acl {
 | 
			
		|||
	char			acl_tag[MAX_ACL_TAG_SIZE];
 | 
			
		||||
	/* Used for PR SPEC_I_PT=1 and REGISTER_AND_MOVE */
 | 
			
		||||
	atomic_t		acl_pr_ref_count;
 | 
			
		||||
	struct se_dev_entry	**device_list;
 | 
			
		||||
	struct hlist_head	lun_entry_hlist;
 | 
			
		||||
	struct se_session	*nacl_sess;
 | 
			
		||||
	struct se_portal_group *se_tpg;
 | 
			
		||||
	spinlock_t		device_list_lock;
 | 
			
		||||
	struct mutex		lun_entry_mutex;
 | 
			
		||||
	spinlock_t		nacl_sess_lock;
 | 
			
		||||
	struct config_group	acl_group;
 | 
			
		||||
	struct config_group	acl_attrib_group;
 | 
			
		||||
| 
						 | 
				
			
			@ -644,20 +642,23 @@ struct se_dev_entry {
 | 
			
		|||
	/* See transport_lunflags_table */
 | 
			
		||||
	u32			lun_flags;
 | 
			
		||||
	u32			mapped_lun;
 | 
			
		||||
	u32			total_cmds;
 | 
			
		||||
	u64			pr_res_key;
 | 
			
		||||
	u64			creation_time;
 | 
			
		||||
	u32			attach_count;
 | 
			
		||||
	u64			read_bytes;
 | 
			
		||||
	u64			write_bytes;
 | 
			
		||||
	atomic_long_t		total_cmds;
 | 
			
		||||
	atomic_long_t		read_bytes;
 | 
			
		||||
	atomic_long_t		write_bytes;
 | 
			
		||||
	atomic_t		ua_count;
 | 
			
		||||
	/* Used for PR SPEC_I_PT=1 and REGISTER_AND_MOVE */
 | 
			
		||||
	atomic_t		pr_ref_count;
 | 
			
		||||
	struct se_lun_acl	*se_lun_acl;
 | 
			
		||||
	struct kref		pr_kref;
 | 
			
		||||
	struct completion	pr_comp;
 | 
			
		||||
	struct se_lun_acl __rcu	*se_lun_acl;
 | 
			
		||||
	spinlock_t		ua_lock;
 | 
			
		||||
	struct se_lun		*se_lun;
 | 
			
		||||
	struct se_lun __rcu	*se_lun;
 | 
			
		||||
	struct list_head	alua_port_list;
 | 
			
		||||
	struct list_head	ua_list;
 | 
			
		||||
	struct hlist_node	link;
 | 
			
		||||
	struct rcu_head		rcu_head;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
struct se_dev_attrib {
 | 
			
		||||
| 
						 | 
				
			
			@ -703,6 +704,7 @@ struct se_port_stat_grps {
 | 
			
		|||
};
 | 
			
		||||
 | 
			
		||||
struct se_lun {
 | 
			
		||||
	u16			lun_rtpi;
 | 
			
		||||
#define SE_LUN_LINK_MAGIC			0xffff7771
 | 
			
		||||
	u32			lun_link_magic;
 | 
			
		||||
	/* See transport_lun_status_table */
 | 
			
		||||
| 
						 | 
				
			
			@ -710,6 +712,7 @@ struct se_lun {
 | 
			
		|||
	u32			lun_access;
 | 
			
		||||
	u32			lun_flags;
 | 
			
		||||
	u32			unpacked_lun;
 | 
			
		||||
	u32			lun_index;
 | 
			
		||||
	atomic_t		lun_acl_count;
 | 
			
		||||
	spinlock_t		lun_acl_lock;
 | 
			
		||||
	spinlock_t		lun_sep_lock;
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
		Loading…
	
		Reference in a new issue