mirror of
https://github.com/Dasharo/linux.git
synced 2026-03-06 15:25:10 -08:00
Merge tag 'nfsd-6.10' of git://git.kernel.org/pub/scm/linux/kernel/git/cel/linux
Pull nfsd updates from Chuck Lever:
"This is a light release containing mostly optimizations, code clean-
ups, and minor bug fixes. This development cycle has focused on non-
upstream kernel work:
1. Continuing to build upstream CI for NFSD, based on kdevops
2. Backporting NFSD filecache-related fixes to selected LTS kernels
One notable new feature in v6.10 NFSD is the addition of a new netlink
protocol dedicated to configuring NFSD. A new user space tool,
nfsdctl, is to be added to nfs-utils. Lots more to come here.
As always I am very grateful to NFSD contributors, reviewers, testers,
and bug reporters who participated during this cycle"
* tag 'nfsd-6.10' of git://git.kernel.org/pub/scm/linux/kernel/git/cel/linux: (29 commits)
NFSD: Force all NFSv4.2 COPY requests to be synchronous
SUNRPC: Fix gss_free_in_token_pages()
NFS/knfsd: Remove the invalid NFS error 'NFSERR_OPNOTSUPP'
knfsd: LOOKUP can return an illegal error value
nfsd: set security label during create operations
NFSD: Add COPY status code to OFFLOAD_STATUS response
NFSD: Record status of async copy operation in struct nfsd4_copy
SUNRPC: Remove comment for sp_lock
NFSD: add listener-{set,get} netlink command
SUNRPC: add a new svc_find_listener helper
SUNRPC: introduce svc_xprt_create_from_sa utility routine
NFSD: add write_version to netlink command
NFSD: convert write_threads to netlink command
NFSD: allow callers to pass in scope string to nfsd_svc
NFSD: move nfsd_mutex handling into nfsd_svc callers
lockd: host: Remove unnecessary statements'host = NULL;'
nfsd: don't create nfsv4recoverydir in nfsdfs when not used.
nfsd: optimise recalculate_deny_mode() for a common case
nfsd: add tracepoint in mark_client_expired_locked
nfsd: new tracepoint for check_slot_seqid
...
This commit is contained in:
@@ -62,6 +62,59 @@ attribute-sets:
|
||||
name: compound-ops
|
||||
type: u32
|
||||
multi-attr: true
|
||||
-
|
||||
name: server
|
||||
attributes:
|
||||
-
|
||||
name: threads
|
||||
type: u32
|
||||
multi-attr: true
|
||||
-
|
||||
name: gracetime
|
||||
type: u32
|
||||
-
|
||||
name: leasetime
|
||||
type: u32
|
||||
-
|
||||
name: scope
|
||||
type: string
|
||||
-
|
||||
name: version
|
||||
attributes:
|
||||
-
|
||||
name: major
|
||||
type: u32
|
||||
-
|
||||
name: minor
|
||||
type: u32
|
||||
-
|
||||
name: enabled
|
||||
type: flag
|
||||
-
|
||||
name: server-proto
|
||||
attributes:
|
||||
-
|
||||
name: version
|
||||
type: nest
|
||||
nested-attributes: version
|
||||
multi-attr: true
|
||||
-
|
||||
name: sock
|
||||
attributes:
|
||||
-
|
||||
name: addr
|
||||
type: binary
|
||||
-
|
||||
name: transport-name
|
||||
type: string
|
||||
-
|
||||
name: server-sock
|
||||
attributes:
|
||||
-
|
||||
name: addr
|
||||
type: nest
|
||||
nested-attributes: sock
|
||||
multi-attr: true
|
||||
|
||||
operations:
|
||||
list:
|
||||
@@ -87,3 +140,60 @@ operations:
|
||||
- sport
|
||||
- dport
|
||||
- compound-ops
|
||||
-
|
||||
name: threads-set
|
||||
doc: set the number of running threads
|
||||
attribute-set: server
|
||||
flags: [ admin-perm ]
|
||||
do:
|
||||
request:
|
||||
attributes:
|
||||
- threads
|
||||
- gracetime
|
||||
- leasetime
|
||||
- scope
|
||||
-
|
||||
name: threads-get
|
||||
doc: get the number of running threads
|
||||
attribute-set: server
|
||||
do:
|
||||
reply:
|
||||
attributes:
|
||||
- threads
|
||||
- gracetime
|
||||
- leasetime
|
||||
- scope
|
||||
-
|
||||
name: version-set
|
||||
doc: set nfs enabled versions
|
||||
attribute-set: server-proto
|
||||
flags: [ admin-perm ]
|
||||
do:
|
||||
request:
|
||||
attributes:
|
||||
- version
|
||||
-
|
||||
name: version-get
|
||||
doc: get nfs enabled versions
|
||||
attribute-set: server-proto
|
||||
do:
|
||||
reply:
|
||||
attributes:
|
||||
- version
|
||||
-
|
||||
name: listener-set
|
||||
doc: set nfs running sockets
|
||||
attribute-set: server-sock
|
||||
flags: [ admin-perm ]
|
||||
do:
|
||||
request:
|
||||
attributes:
|
||||
- addr
|
||||
-
|
||||
name: listener-get
|
||||
doc: get nfs running listeners
|
||||
attribute-set: server-sock
|
||||
do:
|
||||
reply:
|
||||
attributes:
|
||||
- addr
|
||||
|
||||
@@ -117,7 +117,6 @@ static struct nlm_host *nlm_alloc_host(struct nlm_lookup_host_info *ni,
|
||||
if (nsm != NULL)
|
||||
refcount_inc(&nsm->sm_count);
|
||||
else {
|
||||
host = NULL;
|
||||
nsm = nsm_get_handle(ni->net, ni->sap, ni->salen,
|
||||
ni->hostname, ni->hostname_len);
|
||||
if (unlikely(nsm == NULL)) {
|
||||
|
||||
@@ -334,21 +334,25 @@ static void nfsd4_fslocs_free(struct nfsd4_fs_locations *fsloc)
|
||||
static int export_stats_init(struct export_stats *stats)
|
||||
{
|
||||
stats->start_time = ktime_get_seconds();
|
||||
return nfsd_percpu_counters_init(stats->counter, EXP_STATS_COUNTERS_NUM);
|
||||
return percpu_counter_init_many(stats->counter, 0, GFP_KERNEL,
|
||||
EXP_STATS_COUNTERS_NUM);
|
||||
}
|
||||
|
||||
static void export_stats_reset(struct export_stats *stats)
|
||||
{
|
||||
if (stats)
|
||||
nfsd_percpu_counters_reset(stats->counter,
|
||||
EXP_STATS_COUNTERS_NUM);
|
||||
if (stats) {
|
||||
int i;
|
||||
|
||||
for (i = 0; i < EXP_STATS_COUNTERS_NUM; i++)
|
||||
percpu_counter_set(&stats->counter[i], 0);
|
||||
}
|
||||
}
|
||||
|
||||
static void export_stats_destroy(struct export_stats *stats)
|
||||
{
|
||||
if (stats)
|
||||
nfsd_percpu_counters_destroy(stats->counter,
|
||||
EXP_STATS_COUNTERS_NUM);
|
||||
percpu_counter_destroy_many(stats->counter,
|
||||
EXP_STATS_COUNTERS_NUM);
|
||||
}
|
||||
|
||||
static void svc_export_put(struct kref *ref)
|
||||
|
||||
@@ -10,6 +10,36 @@
|
||||
|
||||
#include <uapi/linux/nfsd_netlink.h>
|
||||
|
||||
/* Common nested types */
|
||||
const struct nla_policy nfsd_sock_nl_policy[NFSD_A_SOCK_TRANSPORT_NAME + 1] = {
|
||||
[NFSD_A_SOCK_ADDR] = { .type = NLA_BINARY, },
|
||||
[NFSD_A_SOCK_TRANSPORT_NAME] = { .type = NLA_NUL_STRING, },
|
||||
};
|
||||
|
||||
const struct nla_policy nfsd_version_nl_policy[NFSD_A_VERSION_ENABLED + 1] = {
|
||||
[NFSD_A_VERSION_MAJOR] = { .type = NLA_U32, },
|
||||
[NFSD_A_VERSION_MINOR] = { .type = NLA_U32, },
|
||||
[NFSD_A_VERSION_ENABLED] = { .type = NLA_FLAG, },
|
||||
};
|
||||
|
||||
/* NFSD_CMD_THREADS_SET - do */
|
||||
static const struct nla_policy nfsd_threads_set_nl_policy[NFSD_A_SERVER_SCOPE + 1] = {
|
||||
[NFSD_A_SERVER_THREADS] = { .type = NLA_U32, },
|
||||
[NFSD_A_SERVER_GRACETIME] = { .type = NLA_U32, },
|
||||
[NFSD_A_SERVER_LEASETIME] = { .type = NLA_U32, },
|
||||
[NFSD_A_SERVER_SCOPE] = { .type = NLA_NUL_STRING, },
|
||||
};
|
||||
|
||||
/* NFSD_CMD_VERSION_SET - do */
|
||||
static const struct nla_policy nfsd_version_set_nl_policy[NFSD_A_SERVER_PROTO_VERSION + 1] = {
|
||||
[NFSD_A_SERVER_PROTO_VERSION] = NLA_POLICY_NESTED(nfsd_version_nl_policy),
|
||||
};
|
||||
|
||||
/* NFSD_CMD_LISTENER_SET - do */
|
||||
static const struct nla_policy nfsd_listener_set_nl_policy[NFSD_A_SERVER_SOCK_ADDR + 1] = {
|
||||
[NFSD_A_SERVER_SOCK_ADDR] = NLA_POLICY_NESTED(nfsd_sock_nl_policy),
|
||||
};
|
||||
|
||||
/* Ops table for nfsd */
|
||||
static const struct genl_split_ops nfsd_nl_ops[] = {
|
||||
{
|
||||
@@ -19,6 +49,42 @@ static const struct genl_split_ops nfsd_nl_ops[] = {
|
||||
.done = nfsd_nl_rpc_status_get_done,
|
||||
.flags = GENL_CMD_CAP_DUMP,
|
||||
},
|
||||
{
|
||||
.cmd = NFSD_CMD_THREADS_SET,
|
||||
.doit = nfsd_nl_threads_set_doit,
|
||||
.policy = nfsd_threads_set_nl_policy,
|
||||
.maxattr = NFSD_A_SERVER_SCOPE,
|
||||
.flags = GENL_ADMIN_PERM | GENL_CMD_CAP_DO,
|
||||
},
|
||||
{
|
||||
.cmd = NFSD_CMD_THREADS_GET,
|
||||
.doit = nfsd_nl_threads_get_doit,
|
||||
.flags = GENL_CMD_CAP_DO,
|
||||
},
|
||||
{
|
||||
.cmd = NFSD_CMD_VERSION_SET,
|
||||
.doit = nfsd_nl_version_set_doit,
|
||||
.policy = nfsd_version_set_nl_policy,
|
||||
.maxattr = NFSD_A_SERVER_PROTO_VERSION,
|
||||
.flags = GENL_ADMIN_PERM | GENL_CMD_CAP_DO,
|
||||
},
|
||||
{
|
||||
.cmd = NFSD_CMD_VERSION_GET,
|
||||
.doit = nfsd_nl_version_get_doit,
|
||||
.flags = GENL_CMD_CAP_DO,
|
||||
},
|
||||
{
|
||||
.cmd = NFSD_CMD_LISTENER_SET,
|
||||
.doit = nfsd_nl_listener_set_doit,
|
||||
.policy = nfsd_listener_set_nl_policy,
|
||||
.maxattr = NFSD_A_SERVER_SOCK_ADDR,
|
||||
.flags = GENL_ADMIN_PERM | GENL_CMD_CAP_DO,
|
||||
},
|
||||
{
|
||||
.cmd = NFSD_CMD_LISTENER_GET,
|
||||
.doit = nfsd_nl_listener_get_doit,
|
||||
.flags = GENL_CMD_CAP_DO,
|
||||
},
|
||||
};
|
||||
|
||||
struct genl_family nfsd_nl_family __ro_after_init = {
|
||||
|
||||
@@ -11,11 +11,21 @@
|
||||
|
||||
#include <uapi/linux/nfsd_netlink.h>
|
||||
|
||||
/* Common nested types */
|
||||
extern const struct nla_policy nfsd_sock_nl_policy[NFSD_A_SOCK_TRANSPORT_NAME + 1];
|
||||
extern const struct nla_policy nfsd_version_nl_policy[NFSD_A_VERSION_ENABLED + 1];
|
||||
|
||||
int nfsd_nl_rpc_status_get_start(struct netlink_callback *cb);
|
||||
int nfsd_nl_rpc_status_get_done(struct netlink_callback *cb);
|
||||
|
||||
int nfsd_nl_rpc_status_get_dumpit(struct sk_buff *skb,
|
||||
struct netlink_callback *cb);
|
||||
int nfsd_nl_threads_set_doit(struct sk_buff *skb, struct genl_info *info);
|
||||
int nfsd_nl_threads_get_doit(struct sk_buff *skb, struct genl_info *info);
|
||||
int nfsd_nl_version_set_doit(struct sk_buff *skb, struct genl_info *info);
|
||||
int nfsd_nl_version_get_doit(struct sk_buff *skb, struct genl_info *info);
|
||||
int nfsd_nl_listener_set_doit(struct sk_buff *skb, struct genl_info *info);
|
||||
int nfsd_nl_listener_get_doit(struct sk_buff *skb, struct genl_info *info);
|
||||
|
||||
extern struct genl_family nfsd_nl_family;
|
||||
|
||||
|
||||
@@ -218,6 +218,7 @@ struct nfsd_net {
|
||||
/* Simple check to find out if a given net was properly initialized */
|
||||
#define nfsd_netns_ready(nn) ((nn)->sessionid_hashtbl)
|
||||
|
||||
extern bool nfsd_support_version(int vers);
|
||||
extern void nfsd_netns_free_versions(struct nfsd_net *nn);
|
||||
|
||||
extern unsigned int nfsd_net_id;
|
||||
|
||||
@@ -978,12 +978,12 @@ static int max_cb_time(struct net *net)
|
||||
return max(((u32)nn->nfsd4_lease)/10, 1u) * HZ;
|
||||
}
|
||||
|
||||
static struct workqueue_struct *callback_wq;
|
||||
|
||||
static bool nfsd4_queue_cb(struct nfsd4_callback *cb)
|
||||
{
|
||||
trace_nfsd_cb_queue(cb->cb_clp, cb);
|
||||
return queue_work(callback_wq, &cb->cb_work);
|
||||
struct nfs4_client *clp = cb->cb_clp;
|
||||
|
||||
trace_nfsd_cb_queue(clp, cb);
|
||||
return queue_work(clp->cl_callback_wq, &cb->cb_work);
|
||||
}
|
||||
|
||||
static void nfsd41_cb_inflight_begin(struct nfs4_client *clp)
|
||||
@@ -1153,7 +1153,7 @@ void nfsd4_probe_callback(struct nfs4_client *clp)
|
||||
void nfsd4_probe_callback_sync(struct nfs4_client *clp)
|
||||
{
|
||||
nfsd4_probe_callback(clp);
|
||||
flush_workqueue(callback_wq);
|
||||
flush_workqueue(clp->cl_callback_wq);
|
||||
}
|
||||
|
||||
void nfsd4_change_callback(struct nfs4_client *clp, struct nfs4_cb_conn *conn)
|
||||
@@ -1372,19 +1372,6 @@ static const struct rpc_call_ops nfsd4_cb_ops = {
|
||||
.rpc_release = nfsd4_cb_release,
|
||||
};
|
||||
|
||||
int nfsd4_create_callback_queue(void)
|
||||
{
|
||||
callback_wq = alloc_ordered_workqueue("nfsd4_callbacks", 0);
|
||||
if (!callback_wq)
|
||||
return -ENOMEM;
|
||||
return 0;
|
||||
}
|
||||
|
||||
void nfsd4_destroy_callback_queue(void)
|
||||
{
|
||||
destroy_workqueue(callback_wq);
|
||||
}
|
||||
|
||||
/* must be called under the state lock */
|
||||
void nfsd4_shutdown_callback(struct nfs4_client *clp)
|
||||
{
|
||||
@@ -1398,7 +1385,7 @@ void nfsd4_shutdown_callback(struct nfs4_client *clp)
|
||||
* client, destroy the rpc client, and stop:
|
||||
*/
|
||||
nfsd4_run_cb(&clp->cl_cb_null);
|
||||
flush_workqueue(callback_wq);
|
||||
flush_workqueue(clp->cl_callback_wq);
|
||||
nfsd41_cb_inflight_wait_complete(clp);
|
||||
}
|
||||
|
||||
@@ -1420,9 +1407,9 @@ static struct nfsd4_conn * __nfsd4_find_backchannel(struct nfs4_client *clp)
|
||||
|
||||
/*
|
||||
* Note there isn't a lot of locking in this code; instead we depend on
|
||||
* the fact that it is run from the callback_wq, which won't run two
|
||||
* work items at once. So, for example, callback_wq handles all access
|
||||
* of cl_cb_client and all calls to rpc_create or rpc_shutdown_client.
|
||||
* the fact that it is run from clp->cl_callback_wq, which won't run two
|
||||
* work items at once. So, for example, clp->cl_callback_wq handles all
|
||||
* access of cl_cb_client and all calls to rpc_create or rpc_shutdown_client.
|
||||
*/
|
||||
static void nfsd4_process_cb_update(struct nfsd4_callback *cb)
|
||||
{
|
||||
|
||||
@@ -1737,7 +1737,7 @@ static void cleanup_async_copy(struct nfsd4_copy *copy)
|
||||
nfs4_put_copy(copy);
|
||||
}
|
||||
|
||||
static void nfsd4_send_cb_offload(struct nfsd4_copy *copy, __be32 nfserr)
|
||||
static void nfsd4_send_cb_offload(struct nfsd4_copy *copy)
|
||||
{
|
||||
struct nfsd4_cb_offload *cbo;
|
||||
|
||||
@@ -1747,12 +1747,12 @@ static void nfsd4_send_cb_offload(struct nfsd4_copy *copy, __be32 nfserr)
|
||||
|
||||
memcpy(&cbo->co_res, ©->cp_res, sizeof(copy->cp_res));
|
||||
memcpy(&cbo->co_fh, ©->fh, sizeof(copy->fh));
|
||||
cbo->co_nfserr = nfserr;
|
||||
cbo->co_nfserr = copy->nfserr;
|
||||
|
||||
nfsd4_init_cb(&cbo->co_cb, copy->cp_clp, &nfsd4_cb_offload_ops,
|
||||
NFSPROC4_CLNT_CB_OFFLOAD);
|
||||
trace_nfsd_cb_offload(copy->cp_clp, &cbo->co_res.cb_stateid,
|
||||
&cbo->co_fh, copy->cp_count, nfserr);
|
||||
&cbo->co_fh, copy->cp_count, copy->nfserr);
|
||||
nfsd4_run_cb(&cbo->co_cb);
|
||||
}
|
||||
|
||||
@@ -1766,7 +1766,6 @@ static void nfsd4_send_cb_offload(struct nfsd4_copy *copy, __be32 nfserr)
|
||||
static int nfsd4_do_async_copy(void *data)
|
||||
{
|
||||
struct nfsd4_copy *copy = (struct nfsd4_copy *)data;
|
||||
__be32 nfserr;
|
||||
|
||||
trace_nfsd_copy_do_async(copy);
|
||||
if (nfsd4_ssc_is_inter(copy)) {
|
||||
@@ -1777,24 +1776,25 @@ static int nfsd4_do_async_copy(void *data)
|
||||
if (IS_ERR(filp)) {
|
||||
switch (PTR_ERR(filp)) {
|
||||
case -EBADF:
|
||||
nfserr = nfserr_wrong_type;
|
||||
copy->nfserr = nfserr_wrong_type;
|
||||
break;
|
||||
default:
|
||||
nfserr = nfserr_offload_denied;
|
||||
copy->nfserr = nfserr_offload_denied;
|
||||
}
|
||||
/* ss_mnt will be unmounted by the laundromat */
|
||||
goto do_callback;
|
||||
}
|
||||
nfserr = nfsd4_do_copy(copy, filp, copy->nf_dst->nf_file,
|
||||
false);
|
||||
copy->nfserr = nfsd4_do_copy(copy, filp, copy->nf_dst->nf_file,
|
||||
false);
|
||||
nfsd4_cleanup_inter_ssc(copy->ss_nsui, filp, copy->nf_dst);
|
||||
} else {
|
||||
nfserr = nfsd4_do_copy(copy, copy->nf_src->nf_file,
|
||||
copy->nf_dst->nf_file, false);
|
||||
copy->nfserr = nfsd4_do_copy(copy, copy->nf_src->nf_file,
|
||||
copy->nf_dst->nf_file, false);
|
||||
}
|
||||
|
||||
do_callback:
|
||||
nfsd4_send_cb_offload(copy, nfserr);
|
||||
set_bit(NFSD4_COPY_F_COMPLETED, ©->cp_flags);
|
||||
nfsd4_send_cb_offload(copy);
|
||||
cleanup_async_copy(copy);
|
||||
return 0;
|
||||
}
|
||||
@@ -1807,6 +1807,13 @@ nfsd4_copy(struct svc_rqst *rqstp, struct nfsd4_compound_state *cstate,
|
||||
__be32 status;
|
||||
struct nfsd4_copy *async_copy = NULL;
|
||||
|
||||
/*
|
||||
* Currently, async COPY is not reliable. Force all COPY
|
||||
* requests to be synchronous to avoid client application
|
||||
* hangs waiting for COPY completion.
|
||||
*/
|
||||
nfsd4_copy_set_sync(copy, true);
|
||||
|
||||
copy->cp_clp = cstate->clp;
|
||||
if (nfsd4_ssc_is_inter(copy)) {
|
||||
trace_nfsd_copy_inter(copy);
|
||||
@@ -2003,11 +2010,16 @@ nfsd4_offload_status(struct svc_rqst *rqstp,
|
||||
struct nfsd4_copy *copy;
|
||||
struct nfs4_client *clp = cstate->clp;
|
||||
|
||||
os->completed = false;
|
||||
spin_lock(&clp->async_lock);
|
||||
copy = find_async_copy_locked(clp, &os->stateid);
|
||||
if (copy)
|
||||
if (copy) {
|
||||
os->count = copy->cp_res.wr_bytes_written;
|
||||
else
|
||||
if (test_bit(NFSD4_COPY_F_COMPLETED, ©->cp_flags)) {
|
||||
os->completed = true;
|
||||
os->status = copy->nfserr;
|
||||
}
|
||||
} else
|
||||
status = nfserr_bad_stateid;
|
||||
spin_unlock(&clp->async_lock);
|
||||
|
||||
@@ -2154,6 +2166,29 @@ nfsd4_verify(struct svc_rqst *rqstp, struct nfsd4_compound_state *cstate,
|
||||
return status == nfserr_same ? nfs_ok : status;
|
||||
}
|
||||
|
||||
static __be32
|
||||
nfsd4_get_dir_delegation(struct svc_rqst *rqstp,
|
||||
struct nfsd4_compound_state *cstate,
|
||||
union nfsd4_op_u *u)
|
||||
{
|
||||
struct nfsd4_get_dir_delegation *gdd = &u->get_dir_delegation;
|
||||
|
||||
/*
|
||||
* RFC 8881, section 18.39.3 says:
|
||||
*
|
||||
* "The server may refuse to grant the delegation. In that case, the
|
||||
* server will return NFS4ERR_DIRDELEG_UNAVAIL."
|
||||
*
|
||||
* This is sub-optimal, since it means that the server would need to
|
||||
* abort compound processing just because the delegation wasn't
|
||||
* available. RFC8881bis should change this to allow the server to
|
||||
* return NFS4_OK with a non-fatal status of GDD4_UNAVAIL in this
|
||||
* situation.
|
||||
*/
|
||||
gdd->gddrnf_status = GDD4_UNAVAIL;
|
||||
return nfs_ok;
|
||||
}
|
||||
|
||||
#ifdef CONFIG_NFSD_PNFS
|
||||
static const struct nfsd4_layout_ops *
|
||||
nfsd4_layout_verify(struct svc_export *exp, unsigned int layout_type)
|
||||
@@ -3082,6 +3117,18 @@ static u32 nfsd4_copy_notify_rsize(const struct svc_rqst *rqstp,
|
||||
* sizeof(__be32);
|
||||
}
|
||||
|
||||
static u32 nfsd4_get_dir_delegation_rsize(const struct svc_rqst *rqstp,
|
||||
const struct nfsd4_op *op)
|
||||
{
|
||||
return (op_encode_hdr_size +
|
||||
1 /* gddr_status */ +
|
||||
op_encode_verifier_maxsz +
|
||||
op_encode_stateid_maxsz +
|
||||
2 /* gddr_notification */ +
|
||||
2 /* gddr_child_attributes */ +
|
||||
2 /* gddr_dir_attributes */);
|
||||
}
|
||||
|
||||
#ifdef CONFIG_NFSD_PNFS
|
||||
static u32 nfsd4_getdeviceinfo_rsize(const struct svc_rqst *rqstp,
|
||||
const struct nfsd4_op *op)
|
||||
@@ -3470,6 +3517,12 @@ static const struct nfsd4_operation nfsd4_ops[] = {
|
||||
.op_get_currentstateid = nfsd4_get_freestateid,
|
||||
.op_rsize_bop = nfsd4_only_status_rsize,
|
||||
},
|
||||
[OP_GET_DIR_DELEGATION] = {
|
||||
.op_func = nfsd4_get_dir_delegation,
|
||||
.op_flags = OP_MODIFIES_SOMETHING,
|
||||
.op_name = "OP_GET_DIR_DELEGATION",
|
||||
.op_rsize_bop = nfsd4_get_dir_delegation_rsize,
|
||||
},
|
||||
#ifdef CONFIG_NFSD_PNFS
|
||||
[OP_GETDEVICEINFO] = {
|
||||
.op_func = nfsd4_getdeviceinfo,
|
||||
|
||||
@@ -541,7 +541,7 @@ same_owner_str(struct nfs4_stateowner *sop, struct xdr_netobj *owner)
|
||||
}
|
||||
|
||||
static struct nfs4_openowner *
|
||||
find_openstateowner_str_locked(unsigned int hashval, struct nfsd4_open *open,
|
||||
find_openstateowner_str(unsigned int hashval, struct nfsd4_open *open,
|
||||
struct nfs4_client *clp)
|
||||
{
|
||||
struct nfs4_stateowner *so;
|
||||
@@ -558,18 +558,6 @@ find_openstateowner_str_locked(unsigned int hashval, struct nfsd4_open *open,
|
||||
return NULL;
|
||||
}
|
||||
|
||||
static struct nfs4_openowner *
|
||||
find_openstateowner_str(unsigned int hashval, struct nfsd4_open *open,
|
||||
struct nfs4_client *clp)
|
||||
{
|
||||
struct nfs4_openowner *oo;
|
||||
|
||||
spin_lock(&clp->cl_lock);
|
||||
oo = find_openstateowner_str_locked(hashval, open, clp);
|
||||
spin_unlock(&clp->cl_lock);
|
||||
return oo;
|
||||
}
|
||||
|
||||
static inline u32
|
||||
opaque_hashval(const void *ptr, int nbytes)
|
||||
{
|
||||
@@ -1409,11 +1397,16 @@ static void
|
||||
recalculate_deny_mode(struct nfs4_file *fp)
|
||||
{
|
||||
struct nfs4_ol_stateid *stp;
|
||||
u32 old_deny;
|
||||
|
||||
spin_lock(&fp->fi_lock);
|
||||
old_deny = fp->fi_share_deny;
|
||||
fp->fi_share_deny = 0;
|
||||
list_for_each_entry(stp, &fp->fi_stateids, st_perfile)
|
||||
list_for_each_entry(stp, &fp->fi_stateids, st_perfile) {
|
||||
fp->fi_share_deny |= bmap_to_share_mode(stp->st_deny_bmap);
|
||||
if (fp->fi_share_deny == old_deny)
|
||||
break;
|
||||
}
|
||||
spin_unlock(&fp->fi_lock);
|
||||
}
|
||||
|
||||
@@ -2245,6 +2238,10 @@ static struct nfs4_client *alloc_client(struct xdr_netobj name,
|
||||
GFP_KERNEL);
|
||||
if (!clp->cl_ownerstr_hashtbl)
|
||||
goto err_no_hashtbl;
|
||||
clp->cl_callback_wq = alloc_ordered_workqueue("nfsd4_callbacks", 0);
|
||||
if (!clp->cl_callback_wq)
|
||||
goto err_no_callback_wq;
|
||||
|
||||
for (i = 0; i < OWNER_HASH_SIZE; i++)
|
||||
INIT_LIST_HEAD(&clp->cl_ownerstr_hashtbl[i]);
|
||||
INIT_LIST_HEAD(&clp->cl_sessions);
|
||||
@@ -2267,6 +2264,8 @@ static struct nfs4_client *alloc_client(struct xdr_netobj name,
|
||||
spin_lock_init(&clp->cl_lock);
|
||||
rpc_init_wait_queue(&clp->cl_cb_waitq, "Backchannel slot table");
|
||||
return clp;
|
||||
err_no_callback_wq:
|
||||
kfree(clp->cl_ownerstr_hashtbl);
|
||||
err_no_hashtbl:
|
||||
kfree(clp->cl_name.data);
|
||||
err_no_name:
|
||||
@@ -2280,6 +2279,7 @@ static void __free_client(struct kref *k)
|
||||
struct nfs4_client *clp = container_of(c, struct nfs4_client, cl_nfsdfs);
|
||||
|
||||
free_svc_cred(&clp->cl_cred);
|
||||
destroy_workqueue(clp->cl_callback_wq);
|
||||
kfree(clp->cl_ownerstr_hashtbl);
|
||||
kfree(clp->cl_name.data);
|
||||
kfree(clp->cl_nii_domain.data);
|
||||
@@ -2352,7 +2352,11 @@ unhash_client(struct nfs4_client *clp)
|
||||
|
||||
static __be32 mark_client_expired_locked(struct nfs4_client *clp)
|
||||
{
|
||||
if (atomic_read(&clp->cl_rpc_users))
|
||||
int users = atomic_read(&clp->cl_rpc_users);
|
||||
|
||||
trace_nfsd_mark_client_expired(clp, users);
|
||||
|
||||
if (users)
|
||||
return nfserr_jukebox;
|
||||
unhash_client_locked(clp);
|
||||
return nfs_ok;
|
||||
@@ -3641,12 +3645,8 @@ out_nolock:
|
||||
return status;
|
||||
}
|
||||
|
||||
static __be32
|
||||
check_slot_seqid(u32 seqid, u32 slot_seqid, int slot_inuse)
|
||||
static __be32 check_slot_seqid(u32 seqid, u32 slot_seqid, bool slot_inuse)
|
||||
{
|
||||
dprintk("%s enter. seqid %d slot_seqid %d\n", __func__, seqid,
|
||||
slot_seqid);
|
||||
|
||||
/* The slot is in use, and no response has been sent. */
|
||||
if (slot_inuse) {
|
||||
if (seqid == slot_seqid)
|
||||
@@ -3823,10 +3823,13 @@ nfsd4_create_session(struct svc_rqst *rqstp,
|
||||
}
|
||||
|
||||
/* RFC 8881 Section 18.36.4 Phase 2: Sequence ID processing. */
|
||||
if (conf)
|
||||
if (conf) {
|
||||
cs_slot = &conf->cl_cs_slot;
|
||||
else
|
||||
trace_nfsd_slot_seqid_conf(conf, cr_ses);
|
||||
} else {
|
||||
cs_slot = &unconf->cl_cs_slot;
|
||||
trace_nfsd_slot_seqid_unconf(unconf, cr_ses);
|
||||
}
|
||||
status = check_slot_seqid(cr_ses->seqid, cs_slot->sl_seqid, 0);
|
||||
switch (status) {
|
||||
case nfs_ok:
|
||||
@@ -4221,6 +4224,7 @@ nfsd4_sequence(struct svc_rqst *rqstp, struct nfsd4_compound_state *cstate,
|
||||
* sr_highest_slotid and the sr_target_slot id to maxslots */
|
||||
seq->maxslots = session->se_fchannel.maxreqs;
|
||||
|
||||
trace_nfsd_slot_seqid_sequence(clp, seq, slot);
|
||||
status = check_slot_seqid(seq->seqid, slot->sl_seqid,
|
||||
slot->sl_flags & NFSD4_SLOT_INUSE);
|
||||
if (status == nfserr_replay_cache) {
|
||||
@@ -4662,21 +4666,32 @@ nfsd4_init_leases_net(struct nfsd_net *nn)
|
||||
atomic_set(&nn->nfsd_courtesy_clients, 0);
|
||||
}
|
||||
|
||||
enum rp_lock {
|
||||
RP_UNLOCKED,
|
||||
RP_LOCKED,
|
||||
RP_UNHASHED,
|
||||
};
|
||||
|
||||
static void init_nfs4_replay(struct nfs4_replay *rp)
|
||||
{
|
||||
rp->rp_status = nfserr_serverfault;
|
||||
rp->rp_buflen = 0;
|
||||
rp->rp_buf = rp->rp_ibuf;
|
||||
mutex_init(&rp->rp_mutex);
|
||||
atomic_set(&rp->rp_locked, RP_UNLOCKED);
|
||||
}
|
||||
|
||||
static void nfsd4_cstate_assign_replay(struct nfsd4_compound_state *cstate,
|
||||
struct nfs4_stateowner *so)
|
||||
static int nfsd4_cstate_assign_replay(struct nfsd4_compound_state *cstate,
|
||||
struct nfs4_stateowner *so)
|
||||
{
|
||||
if (!nfsd4_has_session(cstate)) {
|
||||
mutex_lock(&so->so_replay.rp_mutex);
|
||||
wait_var_event(&so->so_replay.rp_locked,
|
||||
atomic_cmpxchg(&so->so_replay.rp_locked,
|
||||
RP_UNLOCKED, RP_LOCKED) != RP_LOCKED);
|
||||
if (atomic_read(&so->so_replay.rp_locked) == RP_UNHASHED)
|
||||
return -EAGAIN;
|
||||
cstate->replay_owner = nfs4_get_stateowner(so);
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
void nfsd4_cstate_clear_replay(struct nfsd4_compound_state *cstate)
|
||||
@@ -4685,7 +4700,8 @@ void nfsd4_cstate_clear_replay(struct nfsd4_compound_state *cstate)
|
||||
|
||||
if (so != NULL) {
|
||||
cstate->replay_owner = NULL;
|
||||
mutex_unlock(&so->so_replay.rp_mutex);
|
||||
atomic_set(&so->so_replay.rp_locked, RP_UNLOCKED);
|
||||
wake_up_var(&so->so_replay.rp_locked);
|
||||
nfs4_put_stateowner(so);
|
||||
}
|
||||
}
|
||||
@@ -4866,34 +4882,46 @@ nfsd4_find_and_lock_existing_open(struct nfs4_file *fp, struct nfsd4_open *open)
|
||||
}
|
||||
|
||||
static struct nfs4_openowner *
|
||||
alloc_init_open_stateowner(unsigned int strhashval, struct nfsd4_open *open,
|
||||
struct nfsd4_compound_state *cstate)
|
||||
find_or_alloc_open_stateowner(unsigned int strhashval, struct nfsd4_open *open,
|
||||
struct nfsd4_compound_state *cstate)
|
||||
{
|
||||
struct nfs4_client *clp = cstate->clp;
|
||||
struct nfs4_openowner *oo, *ret;
|
||||
struct nfs4_openowner *oo, *new = NULL;
|
||||
|
||||
oo = alloc_stateowner(openowner_slab, &open->op_owner, clp);
|
||||
if (!oo)
|
||||
return NULL;
|
||||
oo->oo_owner.so_ops = &openowner_ops;
|
||||
oo->oo_owner.so_is_open_owner = 1;
|
||||
oo->oo_owner.so_seqid = open->op_seqid;
|
||||
oo->oo_flags = 0;
|
||||
if (nfsd4_has_session(cstate))
|
||||
oo->oo_flags |= NFS4_OO_CONFIRMED;
|
||||
oo->oo_time = 0;
|
||||
oo->oo_last_closed_stid = NULL;
|
||||
INIT_LIST_HEAD(&oo->oo_close_lru);
|
||||
retry:
|
||||
spin_lock(&clp->cl_lock);
|
||||
ret = find_openstateowner_str_locked(strhashval, open, clp);
|
||||
if (ret == NULL) {
|
||||
hash_openowner(oo, clp, strhashval);
|
||||
ret = oo;
|
||||
} else
|
||||
nfs4_free_stateowner(&oo->oo_owner);
|
||||
|
||||
oo = find_openstateowner_str(strhashval, open, clp);
|
||||
if (!oo && new) {
|
||||
hash_openowner(new, clp, strhashval);
|
||||
spin_unlock(&clp->cl_lock);
|
||||
return new;
|
||||
}
|
||||
spin_unlock(&clp->cl_lock);
|
||||
return ret;
|
||||
|
||||
if (oo && !(oo->oo_flags & NFS4_OO_CONFIRMED)) {
|
||||
/* Replace unconfirmed owners without checking for replay. */
|
||||
release_openowner(oo);
|
||||
oo = NULL;
|
||||
}
|
||||
if (oo) {
|
||||
if (new)
|
||||
nfs4_free_stateowner(&new->oo_owner);
|
||||
return oo;
|
||||
}
|
||||
|
||||
new = alloc_stateowner(openowner_slab, &open->op_owner, clp);
|
||||
if (!new)
|
||||
return NULL;
|
||||
new->oo_owner.so_ops = &openowner_ops;
|
||||
new->oo_owner.so_is_open_owner = 1;
|
||||
new->oo_owner.so_seqid = open->op_seqid;
|
||||
new->oo_flags = 0;
|
||||
if (nfsd4_has_session(cstate))
|
||||
new->oo_flags |= NFS4_OO_CONFIRMED;
|
||||
new->oo_time = 0;
|
||||
new->oo_last_closed_stid = NULL;
|
||||
INIT_LIST_HEAD(&new->oo_close_lru);
|
||||
goto retry;
|
||||
}
|
||||
|
||||
static struct nfs4_ol_stateid *
|
||||
@@ -4969,7 +4997,11 @@ move_to_close_lru(struct nfs4_ol_stateid *s, struct net *net)
|
||||
* Wait for the refcount to drop to 2. Since it has been unhashed,
|
||||
* there should be no danger of the refcount going back up again at
|
||||
* this point.
|
||||
* Some threads with a reference might be waiting for rp_locked,
|
||||
* so tell them to stop waiting.
|
||||
*/
|
||||
atomic_set(&oo->oo_owner.so_replay.rp_locked, RP_UNHASHED);
|
||||
wake_up_var(&oo->oo_owner.so_replay.rp_locked);
|
||||
wait_event(close_wq, refcount_read(&s->st_stid.sc_count) == 2);
|
||||
|
||||
release_all_access(s);
|
||||
@@ -5342,27 +5374,19 @@ nfsd4_process_open1(struct nfsd4_compound_state *cstate,
|
||||
clp = cstate->clp;
|
||||
|
||||
strhashval = ownerstr_hashval(&open->op_owner);
|
||||
oo = find_openstateowner_str(strhashval, open, clp);
|
||||
retry:
|
||||
oo = find_or_alloc_open_stateowner(strhashval, open, cstate);
|
||||
open->op_openowner = oo;
|
||||
if (!oo) {
|
||||
goto new_owner;
|
||||
}
|
||||
if (!(oo->oo_flags & NFS4_OO_CONFIRMED)) {
|
||||
/* Replace unconfirmed owners without checking for replay. */
|
||||
release_openowner(oo);
|
||||
open->op_openowner = NULL;
|
||||
goto new_owner;
|
||||
if (!oo)
|
||||
return nfserr_jukebox;
|
||||
if (nfsd4_cstate_assign_replay(cstate, &oo->oo_owner) == -EAGAIN) {
|
||||
nfs4_put_stateowner(&oo->oo_owner);
|
||||
goto retry;
|
||||
}
|
||||
status = nfsd4_check_seqid(cstate, &oo->oo_owner, open->op_seqid);
|
||||
if (status)
|
||||
return status;
|
||||
goto alloc_stateid;
|
||||
new_owner:
|
||||
oo = alloc_init_open_stateowner(strhashval, open, cstate);
|
||||
if (oo == NULL)
|
||||
return nfserr_jukebox;
|
||||
open->op_openowner = oo;
|
||||
alloc_stateid:
|
||||
|
||||
open->op_stp = nfs4_alloc_open_stateid(clp);
|
||||
if (!open->op_stp)
|
||||
return nfserr_jukebox;
|
||||
@@ -6133,12 +6157,8 @@ out:
|
||||
void nfsd4_cleanup_open_state(struct nfsd4_compound_state *cstate,
|
||||
struct nfsd4_open *open)
|
||||
{
|
||||
if (open->op_openowner) {
|
||||
struct nfs4_stateowner *so = &open->op_openowner->oo_owner;
|
||||
|
||||
nfsd4_cstate_assign_replay(cstate, so);
|
||||
nfs4_put_stateowner(so);
|
||||
}
|
||||
if (open->op_openowner)
|
||||
nfs4_put_stateowner(&open->op_openowner->oo_owner);
|
||||
if (open->op_file)
|
||||
kmem_cache_free(file_slab, open->op_file);
|
||||
if (open->op_stp)
|
||||
@@ -7202,12 +7222,16 @@ nfs4_preprocess_seqid_op(struct nfsd4_compound_state *cstate, u32 seqid,
|
||||
trace_nfsd_preprocess(seqid, stateid);
|
||||
|
||||
*stpp = NULL;
|
||||
retry:
|
||||
status = nfsd4_lookup_stateid(cstate, stateid,
|
||||
typemask, statusmask, &s, nn);
|
||||
if (status)
|
||||
return status;
|
||||
stp = openlockstateid(s);
|
||||
nfsd4_cstate_assign_replay(cstate, stp->st_stateowner);
|
||||
if (nfsd4_cstate_assign_replay(cstate, stp->st_stateowner) == -EAGAIN) {
|
||||
nfs4_put_stateowner(stp->st_stateowner);
|
||||
goto retry;
|
||||
}
|
||||
|
||||
status = nfs4_seqid_op_checks(cstate, stateid, seqid, stp);
|
||||
if (!status)
|
||||
@@ -7349,7 +7373,7 @@ out:
|
||||
return status;
|
||||
}
|
||||
|
||||
static void nfsd4_close_open_stateid(struct nfs4_ol_stateid *s)
|
||||
static bool nfsd4_close_open_stateid(struct nfs4_ol_stateid *s)
|
||||
{
|
||||
struct nfs4_client *clp = s->st_stid.sc_client;
|
||||
bool unhashed;
|
||||
@@ -7366,11 +7390,11 @@ static void nfsd4_close_open_stateid(struct nfs4_ol_stateid *s)
|
||||
list_for_each_entry(stp, &reaplist, st_locks)
|
||||
nfs4_free_cpntf_statelist(clp->net, &stp->st_stid);
|
||||
free_ol_stateid_reaplist(&reaplist);
|
||||
return false;
|
||||
} else {
|
||||
spin_unlock(&clp->cl_lock);
|
||||
free_ol_stateid_reaplist(&reaplist);
|
||||
if (unhashed)
|
||||
move_to_close_lru(s, clp->net);
|
||||
return unhashed;
|
||||
}
|
||||
}
|
||||
|
||||
@@ -7386,6 +7410,7 @@ nfsd4_close(struct svc_rqst *rqstp, struct nfsd4_compound_state *cstate,
|
||||
struct nfs4_ol_stateid *stp;
|
||||
struct net *net = SVC_NET(rqstp);
|
||||
struct nfsd_net *nn = net_generic(net, nfsd_net_id);
|
||||
bool need_move_to_close_list;
|
||||
|
||||
dprintk("NFSD: nfsd4_close on file %pd\n",
|
||||
cstate->current_fh.fh_dentry);
|
||||
@@ -7410,8 +7435,10 @@ nfsd4_close(struct svc_rqst *rqstp, struct nfsd4_compound_state *cstate,
|
||||
*/
|
||||
nfs4_inc_and_copy_stateid(&close->cl_stateid, &stp->st_stid);
|
||||
|
||||
nfsd4_close_open_stateid(stp);
|
||||
need_move_to_close_list = nfsd4_close_open_stateid(stp);
|
||||
mutex_unlock(&stp->st_mutex);
|
||||
if (need_move_to_close_list)
|
||||
move_to_close_lru(stp, net);
|
||||
|
||||
/* v4.1+ suggests that we send a special stateid in here, since the
|
||||
* clients should just ignore this anyway. Since this is not useful
|
||||
@@ -8625,12 +8652,6 @@ nfs4_state_start(void)
|
||||
if (ret)
|
||||
return ret;
|
||||
|
||||
ret = nfsd4_create_callback_queue();
|
||||
if (ret) {
|
||||
rhltable_destroy(&nfs4_file_rhltable);
|
||||
return ret;
|
||||
}
|
||||
|
||||
set_max_delegations();
|
||||
return 0;
|
||||
}
|
||||
@@ -8671,7 +8692,6 @@ nfs4_state_shutdown_net(struct net *net)
|
||||
void
|
||||
nfs4_state_shutdown(void)
|
||||
{
|
||||
nfsd4_destroy_callback_queue();
|
||||
rhltable_destroy(&nfs4_file_rhltable);
|
||||
}
|
||||
|
||||
|
||||
@@ -1732,6 +1732,35 @@ nfsd4_decode_free_stateid(struct nfsd4_compoundargs *argp,
|
||||
return nfsd4_decode_stateid4(argp, &free_stateid->fr_stateid);
|
||||
}
|
||||
|
||||
static __be32
|
||||
nfsd4_decode_get_dir_delegation(struct nfsd4_compoundargs *argp,
|
||||
union nfsd4_op_u *u)
|
||||
{
|
||||
struct nfsd4_get_dir_delegation *gdd = &u->get_dir_delegation;
|
||||
__be32 status;
|
||||
|
||||
memset(gdd, 0, sizeof(*gdd));
|
||||
|
||||
if (xdr_stream_decode_bool(argp->xdr, &gdd->gdda_signal_deleg_avail) < 0)
|
||||
return nfserr_bad_xdr;
|
||||
status = nfsd4_decode_bitmap4(argp, gdd->gdda_notification_types,
|
||||
ARRAY_SIZE(gdd->gdda_notification_types));
|
||||
if (status)
|
||||
return status;
|
||||
status = nfsd4_decode_nfstime4(argp, &gdd->gdda_child_attr_delay);
|
||||
if (status)
|
||||
return status;
|
||||
status = nfsd4_decode_nfstime4(argp, &gdd->gdda_dir_attr_delay);
|
||||
if (status)
|
||||
return status;
|
||||
status = nfsd4_decode_bitmap4(argp, gdd->gdda_child_attributes,
|
||||
ARRAY_SIZE(gdd->gdda_child_attributes));
|
||||
if (status)
|
||||
return status;
|
||||
return nfsd4_decode_bitmap4(argp, gdd->gdda_dir_attributes,
|
||||
ARRAY_SIZE(gdd->gdda_dir_attributes));
|
||||
}
|
||||
|
||||
#ifdef CONFIG_NFSD_PNFS
|
||||
static __be32
|
||||
nfsd4_decode_getdeviceinfo(struct nfsd4_compoundargs *argp,
|
||||
@@ -2370,7 +2399,7 @@ static const nfsd4_dec nfsd4_dec_ops[] = {
|
||||
[OP_CREATE_SESSION] = nfsd4_decode_create_session,
|
||||
[OP_DESTROY_SESSION] = nfsd4_decode_destroy_session,
|
||||
[OP_FREE_STATEID] = nfsd4_decode_free_stateid,
|
||||
[OP_GET_DIR_DELEGATION] = nfsd4_decode_notsupp,
|
||||
[OP_GET_DIR_DELEGATION] = nfsd4_decode_get_dir_delegation,
|
||||
#ifdef CONFIG_NFSD_PNFS
|
||||
[OP_GETDEVICEINFO] = nfsd4_decode_getdeviceinfo,
|
||||
[OP_GETDEVICELIST] = nfsd4_decode_notsupp,
|
||||
@@ -4963,6 +4992,49 @@ nfsd4_encode_test_stateid(struct nfsd4_compoundres *resp, __be32 nfserr,
|
||||
return nfs_ok;
|
||||
}
|
||||
|
||||
static __be32
|
||||
nfsd4_encode_get_dir_delegation(struct nfsd4_compoundres *resp, __be32 nfserr,
|
||||
union nfsd4_op_u *u)
|
||||
{
|
||||
struct nfsd4_get_dir_delegation *gdd = &u->get_dir_delegation;
|
||||
struct xdr_stream *xdr = resp->xdr;
|
||||
__be32 status = nfserr_resource;
|
||||
|
||||
switch(gdd->gddrnf_status) {
|
||||
case GDD4_OK:
|
||||
if (xdr_stream_encode_u32(xdr, GDD4_OK) != XDR_UNIT)
|
||||
break;
|
||||
status = nfsd4_encode_verifier4(xdr, &gdd->gddr_cookieverf);
|
||||
if (status)
|
||||
break;
|
||||
status = nfsd4_encode_stateid4(xdr, &gdd->gddr_stateid);
|
||||
if (status)
|
||||
break;
|
||||
status = nfsd4_encode_bitmap4(xdr, gdd->gddr_notification[0], 0, 0);
|
||||
if (status)
|
||||
break;
|
||||
status = nfsd4_encode_bitmap4(xdr, gdd->gddr_child_attributes[0],
|
||||
gdd->gddr_child_attributes[1],
|
||||
gdd->gddr_child_attributes[2]);
|
||||
if (status)
|
||||
break;
|
||||
status = nfsd4_encode_bitmap4(xdr, gdd->gddr_dir_attributes[0],
|
||||
gdd->gddr_dir_attributes[1],
|
||||
gdd->gddr_dir_attributes[2]);
|
||||
break;
|
||||
default:
|
||||
pr_warn("nfsd: bad gddrnf_status (%u)\n", gdd->gddrnf_status);
|
||||
gdd->gddrnf_will_signal_deleg_avail = 0;
|
||||
fallthrough;
|
||||
case GDD4_UNAVAIL:
|
||||
if (xdr_stream_encode_u32(xdr, GDD4_UNAVAIL) != XDR_UNIT)
|
||||
break;
|
||||
status = nfsd4_encode_bool(xdr, gdd->gddrnf_will_signal_deleg_avail);
|
||||
break;
|
||||
}
|
||||
return status;
|
||||
}
|
||||
|
||||
#ifdef CONFIG_NFSD_PNFS
|
||||
static __be32
|
||||
nfsd4_encode_device_addr4(struct xdr_stream *xdr,
|
||||
@@ -5199,7 +5271,12 @@ nfsd4_encode_offload_status(struct nfsd4_compoundres *resp, __be32 nfserr,
|
||||
if (nfserr != nfs_ok)
|
||||
return nfserr;
|
||||
/* osr_complete<1> */
|
||||
if (xdr_stream_encode_u32(xdr, 0) != XDR_UNIT)
|
||||
if (os->completed) {
|
||||
if (xdr_stream_encode_u32(xdr, 1) != XDR_UNIT)
|
||||
return nfserr_resource;
|
||||
if (xdr_stream_encode_be32(xdr, os->status) != XDR_UNIT)
|
||||
return nfserr_resource;
|
||||
} else if (xdr_stream_encode_u32(xdr, 0) != XDR_UNIT)
|
||||
return nfserr_resource;
|
||||
return nfs_ok;
|
||||
}
|
||||
@@ -5579,7 +5656,7 @@ static const nfsd4_enc nfsd4_enc_ops[] = {
|
||||
[OP_CREATE_SESSION] = nfsd4_encode_create_session,
|
||||
[OP_DESTROY_SESSION] = nfsd4_encode_noop,
|
||||
[OP_FREE_STATEID] = nfsd4_encode_noop,
|
||||
[OP_GET_DIR_DELEGATION] = nfsd4_encode_noop,
|
||||
[OP_GET_DIR_DELEGATION] = nfsd4_encode_get_dir_delegation,
|
||||
#ifdef CONFIG_NFSD_PNFS
|
||||
[OP_GETDEVICEINFO] = nfsd4_encode_getdeviceinfo,
|
||||
[OP_GETDEVICELIST] = nfsd4_encode_noop,
|
||||
|
||||
526
fs/nfsd/nfsctl.c
526
fs/nfsd/nfsctl.c
File diff suppressed because it is too large
Load Diff
@@ -103,7 +103,7 @@ bool nfssvc_encode_voidres(struct svc_rqst *rqstp,
|
||||
/*
|
||||
* Function prototypes.
|
||||
*/
|
||||
int nfsd_svc(int nrservs, struct net *net, const struct cred *cred);
|
||||
int nfsd_svc(int nrservs, struct net *net, const struct cred *cred, const char *scope);
|
||||
int nfsd_dispatch(struct svc_rqst *rqstp);
|
||||
|
||||
int nfsd_nrthreads(struct net *);
|
||||
@@ -230,7 +230,6 @@ void nfsd_lockd_shutdown(void);
|
||||
#define nfserr_nospc cpu_to_be32(NFSERR_NOSPC)
|
||||
#define nfserr_rofs cpu_to_be32(NFSERR_ROFS)
|
||||
#define nfserr_mlink cpu_to_be32(NFSERR_MLINK)
|
||||
#define nfserr_opnotsupp cpu_to_be32(NFSERR_OPNOTSUPP)
|
||||
#define nfserr_nametoolong cpu_to_be32(NFSERR_NAMETOOLONG)
|
||||
#define nfserr_notempty cpu_to_be32(NFSERR_NOTEMPTY)
|
||||
#define nfserr_dquot cpu_to_be32(NFSERR_DQUOT)
|
||||
|
||||
@@ -573,7 +573,7 @@ fh_compose(struct svc_fh *fhp, struct svc_export *exp, struct dentry *dentry,
|
||||
_fh_update(fhp, exp, dentry);
|
||||
if (fhp->fh_handle.fh_fileid_type == FILEID_INVALID) {
|
||||
fh_put(fhp);
|
||||
return nfserr_opnotsupp;
|
||||
return nfserr_stale;
|
||||
}
|
||||
|
||||
return 0;
|
||||
@@ -599,7 +599,7 @@ fh_update(struct svc_fh *fhp)
|
||||
|
||||
_fh_update(fhp, fhp->fh_export, dentry);
|
||||
if (fhp->fh_handle.fh_fileid_type == FILEID_INVALID)
|
||||
return nfserr_opnotsupp;
|
||||
return nfserr_stale;
|
||||
return 0;
|
||||
out_bad:
|
||||
printk(KERN_ERR "fh_update: fh not verified!\n");
|
||||
|
||||
@@ -133,8 +133,7 @@ struct svc_program nfsd_program = {
|
||||
.pg_rpcbind_set = nfsd_rpcbind_set,
|
||||
};
|
||||
|
||||
static bool
|
||||
nfsd_support_version(int vers)
|
||||
bool nfsd_support_version(int vers)
|
||||
{
|
||||
if (vers >= NFSD_MINVERS && vers < NFSD_NRVERS)
|
||||
return nfsd_version[vers] != NULL;
|
||||
@@ -769,13 +768,14 @@ int nfsd_set_nrthreads(int n, int *nthreads, struct net *net)
|
||||
* this is the first time nrservs is nonzero.
|
||||
*/
|
||||
int
|
||||
nfsd_svc(int nrservs, struct net *net, const struct cred *cred)
|
||||
nfsd_svc(int nrservs, struct net *net, const struct cred *cred, const char *scope)
|
||||
{
|
||||
int error;
|
||||
struct nfsd_net *nn = net_generic(net, nfsd_net_id);
|
||||
struct svc_serv *serv;
|
||||
|
||||
mutex_lock(&nfsd_mutex);
|
||||
lockdep_assert_held(&nfsd_mutex);
|
||||
|
||||
dprintk("nfsd: creating service\n");
|
||||
|
||||
nrservs = max(nrservs, 0);
|
||||
@@ -785,7 +785,7 @@ nfsd_svc(int nrservs, struct net *net, const struct cred *cred)
|
||||
if (nrservs == 0 && nn->nfsd_serv == NULL)
|
||||
goto out;
|
||||
|
||||
strscpy(nn->nfsd_name, utsname()->nodename,
|
||||
strscpy(nn->nfsd_name, scope ? scope : utsname()->nodename,
|
||||
sizeof(nn->nfsd_name));
|
||||
|
||||
error = nfsd_create_serv(net);
|
||||
@@ -804,7 +804,6 @@ out_put:
|
||||
if (serv->sv_nrthreads == 0)
|
||||
nfsd_destroy_serv(net);
|
||||
out:
|
||||
mutex_unlock(&nfsd_mutex);
|
||||
return error;
|
||||
}
|
||||
|
||||
|
||||
@@ -408,6 +408,8 @@ struct nfs4_client {
|
||||
1 << NFSD4_CLIENT_CB_KILL)
|
||||
#define NFSD4_CLIENT_CB_RECALL_ANY (6)
|
||||
unsigned long cl_flags;
|
||||
|
||||
struct workqueue_struct *cl_callback_wq;
|
||||
const struct cred *cl_cb_cred;
|
||||
struct rpc_clnt *cl_cb_client;
|
||||
u32 cl_cb_ident;
|
||||
@@ -486,7 +488,7 @@ struct nfs4_replay {
|
||||
unsigned int rp_buflen;
|
||||
char *rp_buf;
|
||||
struct knfsd_fh rp_openfh;
|
||||
struct mutex rp_mutex;
|
||||
atomic_t rp_locked;
|
||||
char rp_ibuf[NFSD4_REPLAY_ISIZE];
|
||||
};
|
||||
|
||||
@@ -735,8 +737,6 @@ extern void nfsd4_change_callback(struct nfs4_client *clp, struct nfs4_cb_conn *
|
||||
extern void nfsd4_init_cb(struct nfsd4_callback *cb, struct nfs4_client *clp,
|
||||
const struct nfsd4_callback_ops *ops, enum nfsd4_cb_op op);
|
||||
extern bool nfsd4_run_cb(struct nfsd4_callback *cb);
|
||||
extern int nfsd4_create_callback_queue(void);
|
||||
extern void nfsd4_destroy_callback_queue(void);
|
||||
extern void nfsd4_shutdown_callback(struct nfs4_client *);
|
||||
extern void nfsd4_shutdown_copy(struct nfs4_client *clp);
|
||||
extern struct nfs4_client_reclaim *nfs4_client_to_reclaim(struct xdr_netobj name,
|
||||
|
||||
@@ -73,48 +73,6 @@ static int nfsd_show(struct seq_file *seq, void *v)
|
||||
|
||||
DEFINE_PROC_SHOW_ATTRIBUTE(nfsd);
|
||||
|
||||
int nfsd_percpu_counters_init(struct percpu_counter *counters, int num)
|
||||
{
|
||||
int i, err = 0;
|
||||
|
||||
for (i = 0; !err && i < num; i++)
|
||||
err = percpu_counter_init(&counters[i], 0, GFP_KERNEL);
|
||||
|
||||
if (!err)
|
||||
return 0;
|
||||
|
||||
for (; i > 0; i--)
|
||||
percpu_counter_destroy(&counters[i-1]);
|
||||
|
||||
return err;
|
||||
}
|
||||
|
||||
void nfsd_percpu_counters_reset(struct percpu_counter counters[], int num)
|
||||
{
|
||||
int i;
|
||||
|
||||
for (i = 0; i < num; i++)
|
||||
percpu_counter_set(&counters[i], 0);
|
||||
}
|
||||
|
||||
void nfsd_percpu_counters_destroy(struct percpu_counter counters[], int num)
|
||||
{
|
||||
int i;
|
||||
|
||||
for (i = 0; i < num; i++)
|
||||
percpu_counter_destroy(&counters[i]);
|
||||
}
|
||||
|
||||
int nfsd_stat_counters_init(struct nfsd_net *nn)
|
||||
{
|
||||
return nfsd_percpu_counters_init(nn->counter, NFSD_STATS_COUNTERS_NUM);
|
||||
}
|
||||
|
||||
void nfsd_stat_counters_destroy(struct nfsd_net *nn)
|
||||
{
|
||||
nfsd_percpu_counters_destroy(nn->counter, NFSD_STATS_COUNTERS_NUM);
|
||||
}
|
||||
|
||||
void nfsd_proc_stat_init(struct net *net)
|
||||
{
|
||||
struct nfsd_net *nn = net_generic(net, nfsd_net_id);
|
||||
|
||||
@@ -10,11 +10,6 @@
|
||||
#include <uapi/linux/nfsd/stats.h>
|
||||
#include <linux/percpu_counter.h>
|
||||
|
||||
int nfsd_percpu_counters_init(struct percpu_counter *counters, int num);
|
||||
void nfsd_percpu_counters_reset(struct percpu_counter *counters, int num);
|
||||
void nfsd_percpu_counters_destroy(struct percpu_counter *counters, int num);
|
||||
int nfsd_stat_counters_init(struct nfsd_net *nn);
|
||||
void nfsd_stat_counters_destroy(struct nfsd_net *nn);
|
||||
void nfsd_proc_stat_init(struct net *net);
|
||||
void nfsd_proc_stat_shutdown(struct net *net);
|
||||
|
||||
|
||||
100
fs/nfsd/trace.h
100
fs/nfsd/trace.h
@@ -749,6 +749,76 @@ TRACE_EVENT_CONDITION(nfsd_seq4_status,
|
||||
)
|
||||
);
|
||||
|
||||
DECLARE_EVENT_CLASS(nfsd_cs_slot_class,
|
||||
TP_PROTO(
|
||||
const struct nfs4_client *clp,
|
||||
const struct nfsd4_create_session *cs
|
||||
),
|
||||
TP_ARGS(clp, cs),
|
||||
TP_STRUCT__entry(
|
||||
__field(u32, seqid)
|
||||
__field(u32, slot_seqid)
|
||||
__field(u32, cl_boot)
|
||||
__field(u32, cl_id)
|
||||
__sockaddr(addr, clp->cl_cb_conn.cb_addrlen)
|
||||
),
|
||||
TP_fast_assign(
|
||||
const struct nfsd4_clid_slot *slot = &clp->cl_cs_slot;
|
||||
|
||||
__entry->cl_boot = clp->cl_clientid.cl_boot;
|
||||
__entry->cl_id = clp->cl_clientid.cl_id;
|
||||
__assign_sockaddr(addr, &clp->cl_cb_conn.cb_addr,
|
||||
clp->cl_cb_conn.cb_addrlen);
|
||||
__entry->seqid = cs->seqid;
|
||||
__entry->slot_seqid = slot->sl_seqid;
|
||||
),
|
||||
TP_printk("addr=%pISpc client %08x:%08x seqid=%u slot_seqid=%u",
|
||||
__get_sockaddr(addr), __entry->cl_boot, __entry->cl_id,
|
||||
__entry->seqid, __entry->slot_seqid
|
||||
)
|
||||
);
|
||||
|
||||
#define DEFINE_CS_SLOT_EVENT(name) \
|
||||
DEFINE_EVENT(nfsd_cs_slot_class, nfsd_##name, \
|
||||
TP_PROTO( \
|
||||
const struct nfs4_client *clp, \
|
||||
const struct nfsd4_create_session *cs \
|
||||
), \
|
||||
TP_ARGS(clp, cs))
|
||||
|
||||
DEFINE_CS_SLOT_EVENT(slot_seqid_conf);
|
||||
DEFINE_CS_SLOT_EVENT(slot_seqid_unconf);
|
||||
|
||||
TRACE_EVENT(nfsd_slot_seqid_sequence,
|
||||
TP_PROTO(
|
||||
const struct nfs4_client *clp,
|
||||
const struct nfsd4_sequence *seq,
|
||||
const struct nfsd4_slot *slot
|
||||
),
|
||||
TP_ARGS(clp, seq, slot),
|
||||
TP_STRUCT__entry(
|
||||
__field(u32, seqid)
|
||||
__field(u32, slot_seqid)
|
||||
__field(u32, cl_boot)
|
||||
__field(u32, cl_id)
|
||||
__sockaddr(addr, clp->cl_cb_conn.cb_addrlen)
|
||||
__field(bool, in_use)
|
||||
),
|
||||
TP_fast_assign(
|
||||
__entry->cl_boot = clp->cl_clientid.cl_boot;
|
||||
__entry->cl_id = clp->cl_clientid.cl_id;
|
||||
__assign_sockaddr(addr, &clp->cl_cb_conn.cb_addr,
|
||||
clp->cl_cb_conn.cb_addrlen);
|
||||
__entry->seqid = seq->seqid;
|
||||
__entry->slot_seqid = slot->sl_seqid;
|
||||
),
|
||||
TP_printk("addr=%pISpc client %08x:%08x seqid=%u slot_seqid=%u (%sin use)",
|
||||
__get_sockaddr(addr), __entry->cl_boot, __entry->cl_id,
|
||||
__entry->seqid, __entry->slot_seqid,
|
||||
__entry->in_use ? "" : "not "
|
||||
)
|
||||
);
|
||||
|
||||
DECLARE_EVENT_CLASS(nfsd_clientid_class,
|
||||
TP_PROTO(const clientid_t *clid),
|
||||
TP_ARGS(clid),
|
||||
@@ -778,6 +848,30 @@ DEFINE_CLIENTID_EVENT(purged);
|
||||
DEFINE_CLIENTID_EVENT(renew);
|
||||
DEFINE_CLIENTID_EVENT(stale);
|
||||
|
||||
TRACE_EVENT(nfsd_mark_client_expired,
|
||||
TP_PROTO(
|
||||
const struct nfs4_client *clp,
|
||||
int cl_rpc_users
|
||||
),
|
||||
TP_ARGS(clp, cl_rpc_users),
|
||||
TP_STRUCT__entry(
|
||||
__field(int, cl_rpc_users)
|
||||
__field(u32, cl_boot)
|
||||
__field(u32, cl_id)
|
||||
__sockaddr(addr, clp->cl_cb_conn.cb_addrlen)
|
||||
),
|
||||
TP_fast_assign(
|
||||
__entry->cl_rpc_users = cl_rpc_users;
|
||||
__entry->cl_boot = clp->cl_clientid.cl_boot;
|
||||
__entry->cl_id = clp->cl_clientid.cl_id;
|
||||
__assign_sockaddr(addr, &clp->cl_cb_conn.cb_addr,
|
||||
clp->cl_cb_conn.cb_addrlen)
|
||||
),
|
||||
TP_printk("addr=%pISpc client %08x:%08x cl_rpc_users=%d",
|
||||
__get_sockaddr(addr), __entry->cl_boot, __entry->cl_id,
|
||||
__entry->cl_rpc_users)
|
||||
);
|
||||
|
||||
DECLARE_EVENT_CLASS(nfsd_net_class,
|
||||
TP_PROTO(const struct nfsd_net *nn),
|
||||
TP_ARGS(nn),
|
||||
@@ -1534,7 +1628,7 @@ TRACE_EVENT(nfsd_cb_seq_status,
|
||||
__entry->seq_status = cb->cb_seq_status;
|
||||
),
|
||||
TP_printk(SUNRPC_TRACE_TASK_SPECIFIER
|
||||
" sessionid=%08x:%08x:%08x:%08x tk_status=%d seq_status=%d\n",
|
||||
" sessionid=%08x:%08x:%08x:%08x tk_status=%d seq_status=%d",
|
||||
__entry->task_id, __entry->client_id,
|
||||
__entry->cl_boot, __entry->cl_id,
|
||||
__entry->seqno, __entry->reserved,
|
||||
@@ -1573,7 +1667,7 @@ TRACE_EVENT(nfsd_cb_free_slot,
|
||||
__entry->slot_seqno = session->se_cb_seq_nr;
|
||||
),
|
||||
TP_printk(SUNRPC_TRACE_TASK_SPECIFIER
|
||||
" sessionid=%08x:%08x:%08x:%08x new slot seqno=%u\n",
|
||||
" sessionid=%08x:%08x:%08x:%08x new slot seqno=%u",
|
||||
__entry->task_id, __entry->client_id,
|
||||
__entry->cl_boot, __entry->cl_id,
|
||||
__entry->seqno, __entry->reserved,
|
||||
@@ -1978,7 +2072,7 @@ TRACE_EVENT(nfsd_ctl_time,
|
||||
__entry->time = time;
|
||||
__assign_str(name, name);
|
||||
),
|
||||
TP_printk("file=%s time=%d\n",
|
||||
TP_printk("file=%s time=%d",
|
||||
__get_str(name), __entry->time
|
||||
)
|
||||
);
|
||||
|
||||
@@ -1422,7 +1422,7 @@ nfsd_create_setattr(struct svc_rqst *rqstp, struct svc_fh *fhp,
|
||||
* Callers expect new file metadata to be committed even
|
||||
* if the attributes have not changed.
|
||||
*/
|
||||
if (iap->ia_valid)
|
||||
if (nfsd_attrs_valid(attrs))
|
||||
status = nfsd_setattr(rqstp, resfhp, attrs, NULL);
|
||||
else
|
||||
status = nfserrno(commit_metadata(resfhp));
|
||||
|
||||
@@ -60,6 +60,14 @@ static inline void nfsd_attrs_free(struct nfsd_attrs *attrs)
|
||||
posix_acl_release(attrs->na_dpacl);
|
||||
}
|
||||
|
||||
static inline bool nfsd_attrs_valid(struct nfsd_attrs *attrs)
|
||||
{
|
||||
struct iattr *iap = attrs->na_iattr;
|
||||
|
||||
return (iap->ia_valid || (attrs->na_seclabel &&
|
||||
attrs->na_seclabel->len));
|
||||
}
|
||||
|
||||
__be32 nfserrno (int errno);
|
||||
int nfsd_cross_mnt(struct svc_rqst *rqstp, struct dentry **dpp,
|
||||
struct svc_export **expp);
|
||||
|
||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user