On Thu, 2015-04-02 at 01:56 -0700, Christoph Hellwig wrote: > On Wed, Apr 01, 2015 at 10:37:27PM -0700, Nicholas A. Bellinger wrote: > > On Wed, 2015-04-01 at 00:04 -0700, Christoph Hellwig wrote: > > > On Tue, Mar 31, 2015 at 11:51:56PM -0700, Nicholas A. Bellinger wrote: > > > > Since last week, enable/disable device_list code has been converted to > > > > use mempool + call_rcu() and performs the RCU pointer swap in > > > > se_node_acl->lun_entry_hlist[] under se_node_acl->lun_entry_mutex. > > > > > > Why use a mempool there? > > > > Would be nice to guarantee a certain number of new se_dev_entry > > allocations always succeed as existing code expects. > > mempools are for I/O path that make guaranteed progress. While the > callers of core_enable_device_list_for_node are in the control path, > and not in a very deep callstack, fairly shallow below the configfs > interface. This is not a use case for mempools, as there is no need > to guarantee progress in deep I/O callstacks. > I was more concerned about the creation of se_dev_entry for dynamically generated se_node_acl in the login path. Having to back out everything upon se_dev_entry allocation failure is annoying. > > The use of t10_pr_registration->pr_reg_deve is still wrong though, as > > it's held for the lifetime of the pr_reg, and not just while > > pr_ref_count is non zero for REGISTER w/ I_PORT + REGISTER_AND_MOVE > > special cases. > > Seems like we should just take the reference properly. > Not exactly. The reference needs to be taken when se_dev_entry is referenced as a remote I_T_L for the two special PR cases. Otherwise, if it's referenced for normal use it would not be possible to forcefully release se_dev_entry without first finding the registration, which requires taking more locks that I want to avoid in that particular code-path. > > I've already fixed this for v2 by using se_dev_entry->pr_reg for > > signaling when a PR registration is active. > > At least in the version you posted ->pr_reg is just used as a boolean > flag, in which case we might want to keep the real def_pr_registered > flag. > Using se_dev_entry->pr_reg as an RCU pointer seems reasonable to me, considering this value can be updated separate from removing the entire se_dev_entry. So that way, at least core_scsi3_pr_seq_non_holder() will get the correct value under rcu_read_lock() after synchronize_rcu() completes in the update path. > > > The refcount is still there in the good old atomic_t-based version, > > > which is perfectl fine for the PRIN and PROUT subcommands which aren't > > > the I/O fast path. > > > > Agreed the percpu_ref is overkill and percpu_ref_init() is problematic > > because it can still -ENOMEM during se_dev_entry creation.. > > > > Switching to a normal kref with a blocking completion probably makes the > > most sense, in order to avoid potentially spinning on atomic_read() if > > APTPL metadata write-out ends up blocking during the two special PR > > cases. > > The nice part about using the dynamically allocated dev entry is that > the shutdown path doesn't need to be sync. You can just use a normal > kref and don't care when it actually gets freed. One thing I didn't > do in my branch but which would e useful is to rename -> pr_ref > to just ->ref as it's a generic refcount - for now it's only used by > the PR code, but there is nothing specific to it in there. Yes, I've already converted to use dynamically allocated dev entries + call_rcu(). Rename to ->ref sounds fine. Will be posting -v2 over the next days. --nab -- To unsubscribe from this list: send the line "unsubscribe target-devel" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html