diff options
Diffstat (limited to 'fs/nfs/nfs4state.c')
| -rw-r--r-- | fs/nfs/nfs4state.c | 441 | 
1 files changed, 328 insertions, 113 deletions
| diff --git a/fs/nfs/nfs4state.c b/fs/nfs/nfs4state.c index e22862f13564..059c01b67a71 100644 --- a/fs/nfs/nfs4state.c +++ b/fs/nfs/nfs4state.c @@ -154,6 +154,19 @@ struct rpc_cred *nfs4_get_machine_cred_locked(struct nfs_client *clp)  	return cred;  } +static void nfs4_root_machine_cred(struct nfs_client *clp) +{ +	struct rpc_cred *cred, *new; + +	new = rpc_lookup_machine_cred(NULL); +	spin_lock(&clp->cl_lock); +	cred = clp->cl_machine_cred; +	clp->cl_machine_cred = new; +	spin_unlock(&clp->cl_lock); +	if (cred != NULL) +		put_rpccred(cred); +} +  static struct rpc_cred *  nfs4_get_renew_cred_server_locked(struct nfs_server *server)  { @@ -202,32 +215,6 @@ out:  	return cred;  } -#if defined(CONFIG_NFS_V4_1) - -static int nfs41_setup_state_renewal(struct nfs_client *clp) -{ -	int status; -	struct nfs_fsinfo fsinfo; - -	if (!test_bit(NFS_CS_CHECK_LEASE_TIME, &clp->cl_res_state)) { -		nfs4_schedule_state_renewal(clp); -		return 0; -	} - -	status = nfs4_proc_get_lease_time(clp, &fsinfo); -	if (status == 0) { -		/* Update lease time and schedule renewal */ -		spin_lock(&clp->cl_lock); -		clp->cl_lease_time = fsinfo.lease_time * HZ; -		clp->cl_last_renewal = jiffies; -		spin_unlock(&clp->cl_lock); - -		nfs4_schedule_state_renewal(clp); -	} - -	return status; -} -  static void nfs4_end_drain_slot_table(struct nfs4_slot_table *tbl)  {  	if (test_and_clear_bit(NFS4_SLOT_TBL_DRAINING, &tbl->slot_tbl_state)) { @@ -241,27 +228,23 @@ static void nfs4_end_drain_session(struct nfs_client *clp)  {  	struct nfs4_session *ses = clp->cl_session; +	if (clp->cl_slot_tbl) { +		nfs4_end_drain_slot_table(clp->cl_slot_tbl); +		return; +	} +  	if (ses != NULL) {  		nfs4_end_drain_slot_table(&ses->bc_slot_table);  		nfs4_end_drain_slot_table(&ses->fc_slot_table);  	}  } -/* - * Signal state manager thread if session fore channel is drained - */ -void nfs4_slot_tbl_drain_complete(struct nfs4_slot_table *tbl) -{ -	if (nfs4_slot_tbl_draining(tbl)) -		complete(&tbl->complete); -} -  static int nfs4_drain_slot_tbl(struct nfs4_slot_table *tbl)  {  	set_bit(NFS4_SLOT_TBL_DRAINING, &tbl->slot_tbl_state);  	spin_lock(&tbl->slot_tbl_lock);  	if (tbl->highest_used_slotid != NFS4_NO_SLOT) { -		INIT_COMPLETION(tbl->complete); +		reinit_completion(&tbl->complete);  		spin_unlock(&tbl->slot_tbl_lock);  		return wait_for_completion_interruptible(&tbl->complete);  	} @@ -274,6 +257,9 @@ static int nfs4_begin_drain_session(struct nfs_client *clp)  	struct nfs4_session *ses = clp->cl_session;  	int ret = 0; +	if (clp->cl_slot_tbl) +		return nfs4_drain_slot_tbl(clp->cl_slot_tbl); +  	/* back channel */  	ret = nfs4_drain_slot_tbl(&ses->bc_slot_table);  	if (ret) @@ -282,6 +268,32 @@ static int nfs4_begin_drain_session(struct nfs_client *clp)  	return nfs4_drain_slot_tbl(&ses->fc_slot_table);  } +#if defined(CONFIG_NFS_V4_1) + +static int nfs41_setup_state_renewal(struct nfs_client *clp) +{ +	int status; +	struct nfs_fsinfo fsinfo; + +	if (!test_bit(NFS_CS_CHECK_LEASE_TIME, &clp->cl_res_state)) { +		nfs4_schedule_state_renewal(clp); +		return 0; +	} + +	status = nfs4_proc_get_lease_time(clp, &fsinfo); +	if (status == 0) { +		/* Update lease time and schedule renewal */ +		spin_lock(&clp->cl_lock); +		clp->cl_lease_time = fsinfo.lease_time * HZ; +		clp->cl_last_renewal = jiffies; +		spin_unlock(&clp->cl_lock); + +		nfs4_schedule_state_renewal(clp); +	} + +	return status; +} +  static void nfs41_finish_session_reset(struct nfs_client *clp)  {  	clear_bit(NFS4CLNT_LEASE_CONFIRM, &clp->cl_state); @@ -339,62 +351,21 @@ int nfs41_discover_server_trunking(struct nfs_client *clp,  	return nfs41_walk_client_list(clp, result, cred);  } -struct rpc_cred *nfs4_get_exchange_id_cred(struct nfs_client *clp) -{ -	struct rpc_cred *cred; - -	spin_lock(&clp->cl_lock); -	cred = nfs4_get_machine_cred_locked(clp); -	spin_unlock(&clp->cl_lock); -	return cred; -} -  #endif /* CONFIG_NFS_V4_1 */ -static struct rpc_cred * -nfs4_get_setclientid_cred_server(struct nfs_server *server) -{ -	struct nfs_client *clp = server->nfs_client; -	struct rpc_cred *cred = NULL; -	struct nfs4_state_owner *sp; -	struct rb_node *pos; - -	spin_lock(&clp->cl_lock); -	pos = rb_first(&server->state_owners); -	if (pos != NULL) { -		sp = rb_entry(pos, struct nfs4_state_owner, so_server_node); -		cred = get_rpccred(sp->so_cred); -	} -	spin_unlock(&clp->cl_lock); -	return cred; -} -  /** - * nfs4_get_setclientid_cred - Acquire credential for a setclientid operation + * nfs4_get_clid_cred - Acquire credential for a setclientid operation   * @clp: client state handle   *   * Returns an rpc_cred with reference count bumped, or NULL.   */ -struct rpc_cred *nfs4_get_setclientid_cred(struct nfs_client *clp) +struct rpc_cred *nfs4_get_clid_cred(struct nfs_client *clp)  { -	struct nfs_server *server;  	struct rpc_cred *cred;  	spin_lock(&clp->cl_lock);  	cred = nfs4_get_machine_cred_locked(clp);  	spin_unlock(&clp->cl_lock); -	if (cred != NULL) -		goto out; - -	rcu_read_lock(); -	list_for_each_entry_rcu(server, &clp->cl_superblocks, client_link) { -		cred = nfs4_get_setclientid_cred_server(server); -		if (cred != NULL) -			break; -	} -	rcu_read_unlock(); - -out:  	return cred;  } @@ -998,7 +969,9 @@ static int nfs4_copy_lock_stateid(nfs4_stateid *dst,  	fl_pid = lockowner->l_pid;  	spin_lock(&state->state_lock);  	lsp = __nfs4_find_lock_state(state, fl_owner, fl_pid, NFS4_ANY_LOCK_TYPE); -	if (lsp != NULL && test_bit(NFS_LOCK_INITIALIZED, &lsp->ls_flags) != 0) { +	if (lsp && test_bit(NFS_LOCK_LOST, &lsp->ls_flags)) +		ret = -EIO; +	else if (lsp != NULL && test_bit(NFS_LOCK_INITIALIZED, &lsp->ls_flags) != 0) {  		nfs4_stateid_copy(dst, &lsp->ls_stateid);  		ret = 0;  		smp_rmb(); @@ -1038,11 +1011,17 @@ static int nfs4_copy_open_stateid(nfs4_stateid *dst, struct nfs4_state *state)  int nfs4_select_rw_stateid(nfs4_stateid *dst, struct nfs4_state *state,  		fmode_t fmode, const struct nfs_lockowner *lockowner)  { -	int ret = 0; +	int ret = nfs4_copy_lock_stateid(dst, state, lockowner); +	if (ret == -EIO) +		/* A lost lock - don't even consider delegations */ +		goto out;  	if (nfs4_copy_delegation_stateid(dst, state->inode, fmode))  		goto out; -	ret = nfs4_copy_lock_stateid(dst, state, lockowner);  	if (ret != -ENOENT) +		/* nfs4_copy_delegation_stateid() didn't over-write +		 * dst, so it still has the lock stateid which we now +		 * choose to use. +		 */  		goto out;  	ret = nfs4_copy_open_stateid(dst, state);  out: @@ -1218,20 +1197,74 @@ void nfs4_schedule_lease_recovery(struct nfs_client *clp)  }  EXPORT_SYMBOL_GPL(nfs4_schedule_lease_recovery); +/** + * nfs4_schedule_migration_recovery - trigger migration recovery + * + * @server: FSID that is migrating + * + * Returns zero if recovery has started, otherwise a negative NFS4ERR + * value is returned. + */ +int nfs4_schedule_migration_recovery(const struct nfs_server *server) +{ +	struct nfs_client *clp = server->nfs_client; + +	if (server->fh_expire_type != NFS4_FH_PERSISTENT) { +		pr_err("NFS: volatile file handles not supported (server %s)\n", +				clp->cl_hostname); +		return -NFS4ERR_IO; +	} + +	if (test_bit(NFS_MIG_FAILED, &server->mig_status)) +		return -NFS4ERR_IO; + +	dprintk("%s: scheduling migration recovery for (%llx:%llx) on %s\n", +			__func__, +			(unsigned long long)server->fsid.major, +			(unsigned long long)server->fsid.minor, +			clp->cl_hostname); + +	set_bit(NFS_MIG_IN_TRANSITION, +			&((struct nfs_server *)server)->mig_status); +	set_bit(NFS4CLNT_MOVED, &clp->cl_state); + +	nfs4_schedule_state_manager(clp); +	return 0; +} +EXPORT_SYMBOL_GPL(nfs4_schedule_migration_recovery); + +/** + * nfs4_schedule_lease_moved_recovery - start lease-moved recovery + * + * @clp: server to check for moved leases + * + */ +void nfs4_schedule_lease_moved_recovery(struct nfs_client *clp) +{ +	dprintk("%s: scheduling lease-moved recovery for client ID %llx on %s\n", +		__func__, clp->cl_clientid, clp->cl_hostname); + +	set_bit(NFS4CLNT_LEASE_MOVED, &clp->cl_state); +	nfs4_schedule_state_manager(clp); +} +EXPORT_SYMBOL_GPL(nfs4_schedule_lease_moved_recovery); +  int nfs4_wait_clnt_recover(struct nfs_client *clp)  {  	int res;  	might_sleep(); +	atomic_inc(&clp->cl_count);  	res = wait_on_bit(&clp->cl_state, NFS4CLNT_MANAGER_RUNNING,  			nfs_wait_bit_killable, TASK_KILLABLE);  	if (res) -		return res; - +		goto out;  	if (clp->cl_cons_state < 0) -		return clp->cl_cons_state; -	return 0; +		res = clp->cl_cons_state; +out: +	nfs_put_client(clp); +	return res;  }  int nfs4_client_recover_expired_lease(struct nfs_client *clp) @@ -1396,8 +1429,8 @@ static int nfs4_reclaim_locks(struct nfs4_state *state, const struct nfs4_state_  			case -NFS4ERR_CONN_NOT_BOUND_TO_SESSION:  				goto out;  			default: -				printk(KERN_ERR "NFS: %s: unhandled error %d. " -					"Zeroing state\n", __func__, status); +				printk(KERN_ERR "NFS: %s: unhandled error %d\n", +					 __func__, status);  			case -ENOMEM:  			case -NFS4ERR_DENIED:  			case -NFS4ERR_RECLAIM_BAD: @@ -1443,14 +1476,16 @@ restart:  		if (status >= 0) {  			status = nfs4_reclaim_locks(state, ops);  			if (status >= 0) { -				spin_lock(&state->state_lock); -				list_for_each_entry(lock, &state->lock_states, ls_locks) { -					if (!test_bit(NFS_LOCK_INITIALIZED, &lock->ls_flags)) -						pr_warn_ratelimited("NFS: " -							"%s: Lock reclaim " -							"failed!\n", __func__); +				if (!test_bit(NFS_DELEGATED_STATE, &state->flags)) { +					spin_lock(&state->state_lock); +					list_for_each_entry(lock, &state->lock_states, ls_locks) { +						if (!test_bit(NFS_LOCK_INITIALIZED, &lock->ls_flags)) +							pr_warn_ratelimited("NFS: " +									    "%s: Lock reclaim " +									    "failed!\n", __func__); +					} +					spin_unlock(&state->state_lock);  				} -				spin_unlock(&state->state_lock);  				nfs4_put_open_state(state);  				spin_lock(&sp->so_lock);  				goto restart; @@ -1458,15 +1493,12 @@ restart:  		}  		switch (status) {  			default: -				printk(KERN_ERR "NFS: %s: unhandled error %d. " -					"Zeroing state\n", __func__, status); +				printk(KERN_ERR "NFS: %s: unhandled error %d\n", +					__func__, status);  			case -ENOENT:  			case -ENOMEM:  			case -ESTALE: -				/* -				 * Open state on this file cannot be recovered -				 * All we can do is revert to using the zero stateid. -				 */ +				/* Open state on this file cannot be recovered */  				nfs4_state_mark_recovery_failed(state, status);  				break;  			case -EAGAIN: @@ -1618,7 +1650,7 @@ static void nfs4_state_end_reclaim_reboot(struct nfs_client *clp)  	if (!nfs4_state_clear_reclaim_reboot(clp))  		return;  	ops = clp->cl_mvops->reboot_recovery_ops; -	cred = ops->get_clid_cred(clp); +	cred = nfs4_get_clid_cred(clp);  	nfs4_reclaim_complete(clp, ops, cred);  	put_rpccred(cred);  } @@ -1647,7 +1679,6 @@ static int nfs4_recovery_handle_error(struct nfs_client *clp, int error)  			nfs4_state_end_reclaim_reboot(clp);  			break;  		case -NFS4ERR_STALE_CLIENTID: -		case -NFS4ERR_LEASE_MOVED:  			set_bit(NFS4CLNT_LEASE_EXPIRED, &clp->cl_state);  			nfs4_state_clear_reclaim_reboot(clp);  			nfs4_state_start_reclaim_reboot(clp); @@ -1732,7 +1763,7 @@ static int nfs4_check_lease(struct nfs_client *clp)  	cred = ops->get_state_renewal_cred_locked(clp);  	spin_unlock(&clp->cl_lock);  	if (cred == NULL) { -		cred = nfs4_get_setclientid_cred(clp); +		cred = nfs4_get_clid_cred(clp);  		status = -ENOKEY;  		if (cred == NULL)  			goto out; @@ -1804,7 +1835,7 @@ static int nfs4_establish_lease(struct nfs_client *clp)  		clp->cl_mvops->reboot_recovery_ops;  	int status; -	cred = ops->get_clid_cred(clp); +	cred = nfs4_get_clid_cred(clp);  	if (cred == NULL)  		return -ENOENT;  	status = ops->establish_clid(clp, cred); @@ -1848,6 +1879,168 @@ static int nfs4_purge_lease(struct nfs_client *clp)  	return 0;  } +/* + * Try remote migration of one FSID from a source server to a + * destination server.  The source server provides a list of + * potential destinations. + * + * Returns zero or a negative NFS4ERR status code. + */ +static int nfs4_try_migration(struct nfs_server *server, struct rpc_cred *cred) +{ +	struct nfs_client *clp = server->nfs_client; +	struct nfs4_fs_locations *locations = NULL; +	struct inode *inode; +	struct page *page; +	int status, result; + +	dprintk("--> %s: FSID %llx:%llx on \"%s\"\n", __func__, +			(unsigned long long)server->fsid.major, +			(unsigned long long)server->fsid.minor, +			clp->cl_hostname); + +	result = 0; +	page = alloc_page(GFP_KERNEL); +	locations = kmalloc(sizeof(struct nfs4_fs_locations), GFP_KERNEL); +	if (page == NULL || locations == NULL) { +		dprintk("<-- %s: no memory\n", __func__); +		goto out; +	} + +	inode = server->super->s_root->d_inode; +	result = nfs4_proc_get_locations(inode, locations, page, cred); +	if (result) { +		dprintk("<-- %s: failed to retrieve fs_locations: %d\n", +			__func__, result); +		goto out; +	} + +	result = -NFS4ERR_NXIO; +	if (!(locations->fattr.valid & NFS_ATTR_FATTR_V4_LOCATIONS)) { +		dprintk("<-- %s: No fs_locations data, migration skipped\n", +			__func__); +		goto out; +	} + +	nfs4_begin_drain_session(clp); + +	status = nfs4_replace_transport(server, locations); +	if (status != 0) { +		dprintk("<-- %s: failed to replace transport: %d\n", +			__func__, status); +		goto out; +	} + +	result = 0; +	dprintk("<-- %s: migration succeeded\n", __func__); + +out: +	if (page != NULL) +		__free_page(page); +	kfree(locations); +	if (result) { +		pr_err("NFS: migration recovery failed (server %s)\n", +				clp->cl_hostname); +		set_bit(NFS_MIG_FAILED, &server->mig_status); +	} +	return result; +} + +/* + * Returns zero or a negative NFS4ERR status code. + */ +static int nfs4_handle_migration(struct nfs_client *clp) +{ +	const struct nfs4_state_maintenance_ops *ops = +				clp->cl_mvops->state_renewal_ops; +	struct nfs_server *server; +	struct rpc_cred *cred; + +	dprintk("%s: migration reported on \"%s\"\n", __func__, +			clp->cl_hostname); + +	spin_lock(&clp->cl_lock); +	cred = ops->get_state_renewal_cred_locked(clp); +	spin_unlock(&clp->cl_lock); +	if (cred == NULL) +		return -NFS4ERR_NOENT; + +	clp->cl_mig_gen++; +restart: +	rcu_read_lock(); +	list_for_each_entry_rcu(server, &clp->cl_superblocks, client_link) { +		int status; + +		if (server->mig_gen == clp->cl_mig_gen) +			continue; +		server->mig_gen = clp->cl_mig_gen; + +		if (!test_and_clear_bit(NFS_MIG_IN_TRANSITION, +						&server->mig_status)) +			continue; + +		rcu_read_unlock(); +		status = nfs4_try_migration(server, cred); +		if (status < 0) { +			put_rpccred(cred); +			return status; +		} +		goto restart; +	} +	rcu_read_unlock(); +	put_rpccred(cred); +	return 0; +} + +/* + * Test each nfs_server on the clp's cl_superblocks list to see + * if it's moved to another server.  Stop when the server no longer + * returns NFS4ERR_LEASE_MOVED. + */ +static int nfs4_handle_lease_moved(struct nfs_client *clp) +{ +	const struct nfs4_state_maintenance_ops *ops = +				clp->cl_mvops->state_renewal_ops; +	struct nfs_server *server; +	struct rpc_cred *cred; + +	dprintk("%s: lease moved reported on \"%s\"\n", __func__, +			clp->cl_hostname); + +	spin_lock(&clp->cl_lock); +	cred = ops->get_state_renewal_cred_locked(clp); +	spin_unlock(&clp->cl_lock); +	if (cred == NULL) +		return -NFS4ERR_NOENT; + +	clp->cl_mig_gen++; +restart: +	rcu_read_lock(); +	list_for_each_entry_rcu(server, &clp->cl_superblocks, client_link) { +		struct inode *inode; +		int status; + +		if (server->mig_gen == clp->cl_mig_gen) +			continue; +		server->mig_gen = clp->cl_mig_gen; + +		rcu_read_unlock(); + +		inode = server->super->s_root->d_inode; +		status = nfs4_proc_fsid_present(inode, cred); +		if (status != -NFS4ERR_MOVED) +			goto restart;	/* wasn't this one */ +		if (nfs4_try_migration(server, cred) == -NFS4ERR_LEASE_MOVED) +			goto restart;	/* there are more */ +		goto out; +	} +	rcu_read_unlock(); + +out: +	put_rpccred(cred); +	return 0; +} +  /**   * nfs4_discover_server_trunking - Detect server IP address trunking   * @@ -1878,7 +2071,7 @@ int nfs4_discover_server_trunking(struct nfs_client *clp,  	mutex_lock(&nfs_clid_init_mutex);  again:  	status  = -ENOENT; -	cred = ops->get_clid_cred(clp); +	cred = nfs4_get_clid_cred(clp);  	if (cred == NULL)  		goto out_unlock; @@ -1896,10 +2089,19 @@ again:  			__func__, status);  		goto again;  	case -EACCES: -		if (i++) +		if (i++ == 0) { +			nfs4_root_machine_cred(clp); +			goto again; +		} +		if (clnt->cl_auth->au_flavor == RPC_AUTH_UNIX)  			break;  	case -NFS4ERR_CLID_INUSE:  	case -NFS4ERR_WRONGSEC: +		/* No point in retrying if we already used RPC_AUTH_UNIX */ +		if (clnt->cl_auth->au_flavor == RPC_AUTH_UNIX) { +			status = -EPERM; +			break; +		}  		clnt = rpc_clone_client_set_auth(clnt, RPC_AUTH_UNIX);  		if (IS_ERR(clnt)) {  			status = PTR_ERR(clnt); @@ -2032,9 +2234,10 @@ void nfs41_handle_sequence_flag_errors(struct nfs_client *clp, u32 flags)  		nfs41_handle_server_reboot(clp);  	if (flags & (SEQ4_STATUS_EXPIRED_ALL_STATE_REVOKED |  			    SEQ4_STATUS_EXPIRED_SOME_STATE_REVOKED | -			    SEQ4_STATUS_ADMIN_STATE_REVOKED | -			    SEQ4_STATUS_LEASE_MOVED)) +			    SEQ4_STATUS_ADMIN_STATE_REVOKED))  		nfs41_handle_state_revoked(clp); +	if (flags & SEQ4_STATUS_LEASE_MOVED) +		nfs4_schedule_lease_moved_recovery(clp);  	if (flags & SEQ4_STATUS_RECALLABLE_STATE_REVOKED)  		nfs41_handle_recallable_state_revoked(clp);  	if (flags & SEQ4_STATUS_BACKCHANNEL_FAULT) @@ -2052,7 +2255,7 @@ static int nfs4_reset_session(struct nfs_client *clp)  	if (!nfs4_has_session(clp))  		return 0;  	nfs4_begin_drain_session(clp); -	cred = nfs4_get_exchange_id_cred(clp); +	cred = nfs4_get_clid_cred(clp);  	status = nfs4_proc_destroy_session(clp->cl_session, cred);  	switch (status) {  	case 0: @@ -2095,7 +2298,7 @@ static int nfs4_bind_conn_to_session(struct nfs_client *clp)  	if (!nfs4_has_session(clp))  		return 0;  	nfs4_begin_drain_session(clp); -	cred = nfs4_get_exchange_id_cred(clp); +	cred = nfs4_get_clid_cred(clp);  	ret = nfs4_proc_bind_conn_to_session(clp, cred);  	if (cred)  		put_rpccred(cred); @@ -2116,7 +2319,6 @@ static int nfs4_bind_conn_to_session(struct nfs_client *clp)  }  #else /* CONFIG_NFS_V4_1 */  static int nfs4_reset_session(struct nfs_client *clp) { return 0; } -static int nfs4_end_drain_session(struct nfs_client *clp) { return 0; }  static int nfs4_bind_conn_to_session(struct nfs_client *clp)  { @@ -2173,7 +2375,20 @@ static void nfs4_state_manager(struct nfs_client *clp)  			status = nfs4_check_lease(clp);  			if (status < 0)  				goto out_error; -			continue; +		} + +		if (test_and_clear_bit(NFS4CLNT_MOVED, &clp->cl_state)) { +			section = "migration"; +			status = nfs4_handle_migration(clp); +			if (status < 0) +				goto out_error; +		} + +		if (test_and_clear_bit(NFS4CLNT_LEASE_MOVED, &clp->cl_state)) { +			section = "lease moved"; +			status = nfs4_handle_lease_moved(clp); +			if (status < 0) +				goto out_error;  		}  		/* First recover reboot state... */ | 
