afs: Use the per-peer app data provided by rxrpc

Make use of the per-peer application data that rxrpc now allows the
application to store on the rxrpc_peer struct to hold a back pointer to the
afs_server record that peer represents an endpoint for.

Then, when a call comes in to the AFS cache manager, this can be used to
map it to the correct server record rather than having to use a
UUID-to-server mapping table and having to do an additional lookup.

Signed-off-by: David Howells <dhowells@redhat.com>
cc: Marc Dionne <marc.dionne@auristor.com>
cc: linux-afs@lists.infradead.org
cc: linux-fsdevel@vger.kernel.org
Link: https://lore.kernel.org/r/20250224234154.2014840-14-dhowells@redhat.com/ # v1
Link: https://lore.kernel.org/r/20250310094206.801057-10-dhowells@redhat.com/ # v4
pull/1113/merge
David Howells 2025-01-23 11:01:55 +00:00
parent f3a123b254
commit 40e8b52fe8
9 changed files with 120 additions and 123 deletions

View File

@ -362,3 +362,53 @@ int afs_merge_fs_addr6(struct afs_net *net, struct afs_addr_list *alist,
alist->nr_addrs++; alist->nr_addrs++;
return 0; return 0;
} }
/*
* Set the app data on the rxrpc peers an address list points to
*/
void afs_set_peer_appdata(struct afs_server *server,
struct afs_addr_list *old_alist,
struct afs_addr_list *new_alist)
{
unsigned long data = (unsigned long)server;
int n = 0, o = 0;
if (!old_alist) {
/* New server. Just set all. */
for (; n < new_alist->nr_addrs; n++)
rxrpc_kernel_set_peer_data(new_alist->addrs[n].peer, data);
return;
}
if (!new_alist) {
/* Dead server. Just remove all. */
for (; o < old_alist->nr_addrs; o++)
rxrpc_kernel_set_peer_data(old_alist->addrs[o].peer, 0);
return;
}
/* Walk through the two lists simultaneously, setting new peers and
* clearing old ones. The two lists are ordered by pointer to peer
* record.
*/
while (n < new_alist->nr_addrs && o < old_alist->nr_addrs) {
struct rxrpc_peer *pn = new_alist->addrs[n].peer;
struct rxrpc_peer *po = old_alist->addrs[o].peer;
if (pn == po)
continue;
if (pn < po) {
rxrpc_kernel_set_peer_data(pn, data);
n++;
} else {
rxrpc_kernel_set_peer_data(po, 0);
o++;
}
}
if (n < new_alist->nr_addrs)
for (; n < new_alist->nr_addrs; n++)
rxrpc_kernel_set_peer_data(new_alist->addrs[n].peer, data);
if (o < old_alist->nr_addrs)
for (; o < old_alist->nr_addrs; o++)
rxrpc_kernel_set_peer_data(old_alist->addrs[o].peer, 0);
}

View File

@ -138,49 +138,6 @@ bool afs_cm_incoming_call(struct afs_call *call)
} }
} }
/*
* Find the server record by peer address and record a probe to the cache
* manager from a server.
*/
static int afs_find_cm_server_by_peer(struct afs_call *call)
{
struct sockaddr_rxrpc srx;
struct afs_server *server;
struct rxrpc_peer *peer;
peer = rxrpc_kernel_get_call_peer(call->net->socket, call->rxcall);
server = afs_find_server(call->net, peer);
if (!server) {
trace_afs_cm_no_server(call, &srx);
return 0;
}
call->server = server;
return 0;
}
/*
* Find the server record by server UUID and record a probe to the cache
* manager from a server.
*/
static int afs_find_cm_server_by_uuid(struct afs_call *call,
struct afs_uuid *uuid)
{
struct afs_server *server;
rcu_read_lock();
server = afs_find_server_by_uuid(call->net, call->request);
rcu_read_unlock();
if (!server) {
trace_afs_cm_no_server_u(call, call->request);
return 0;
}
call->server = server;
return 0;
}
/* /*
* Clean up a cache manager call. * Clean up a cache manager call.
*/ */
@ -322,10 +279,7 @@ static int afs_deliver_cb_callback(struct afs_call *call)
if (!afs_check_call_state(call, AFS_CALL_SV_REPLYING)) if (!afs_check_call_state(call, AFS_CALL_SV_REPLYING))
return afs_io_error(call, afs_io_error_cm_reply); return afs_io_error(call, afs_io_error_cm_reply);
return 0;
/* we'll need the file server record as that tells us which set of
* vnodes to operate upon */
return afs_find_cm_server_by_peer(call);
} }
/* /*
@ -349,18 +303,10 @@ static void SRXAFSCB_InitCallBackState(struct work_struct *work)
*/ */
static int afs_deliver_cb_init_call_back_state(struct afs_call *call) static int afs_deliver_cb_init_call_back_state(struct afs_call *call)
{ {
int ret;
_enter(""); _enter("");
afs_extract_discard(call, 0); afs_extract_discard(call, 0);
ret = afs_extract_data(call, false); return afs_extract_data(call, false);
if (ret < 0)
return ret;
/* we'll need the file server record as that tells us which set of
* vnodes to operate upon */
return afs_find_cm_server_by_peer(call);
} }
/* /*
@ -373,8 +319,6 @@ static int afs_deliver_cb_init_call_back_state3(struct afs_call *call)
__be32 *b; __be32 *b;
int ret; int ret;
_enter("");
_enter("{%u}", call->unmarshall); _enter("{%u}", call->unmarshall);
switch (call->unmarshall) { switch (call->unmarshall) {
@ -421,9 +365,13 @@ static int afs_deliver_cb_init_call_back_state3(struct afs_call *call)
if (!afs_check_call_state(call, AFS_CALL_SV_REPLYING)) if (!afs_check_call_state(call, AFS_CALL_SV_REPLYING))
return afs_io_error(call, afs_io_error_cm_reply); return afs_io_error(call, afs_io_error_cm_reply);
/* we'll need the file server record as that tells us which set of if (memcmp(call->request, &call->server->_uuid, sizeof(call->server->_uuid)) != 0) {
* vnodes to operate upon */ pr_notice("Callback UUID does not match fileserver UUID\n");
return afs_find_cm_server_by_uuid(call, call->request); trace_afs_cm_no_server_u(call, call->request);
return 0;
}
return 0;
} }
/* /*
@ -455,7 +403,7 @@ static int afs_deliver_cb_probe(struct afs_call *call)
if (!afs_check_call_state(call, AFS_CALL_SV_REPLYING)) if (!afs_check_call_state(call, AFS_CALL_SV_REPLYING))
return afs_io_error(call, afs_io_error_cm_reply); return afs_io_error(call, afs_io_error_cm_reply);
return afs_find_cm_server_by_peer(call); return 0;
} }
/* /*
@ -533,7 +481,7 @@ static int afs_deliver_cb_probe_uuid(struct afs_call *call)
if (!afs_check_call_state(call, AFS_CALL_SV_REPLYING)) if (!afs_check_call_state(call, AFS_CALL_SV_REPLYING))
return afs_io_error(call, afs_io_error_cm_reply); return afs_io_error(call, afs_io_error_cm_reply);
return afs_find_cm_server_by_peer(call); return 0;
} }
/* /*
@ -593,7 +541,7 @@ static int afs_deliver_cb_tell_me_about_yourself(struct afs_call *call)
if (!afs_check_call_state(call, AFS_CALL_SV_REPLYING)) if (!afs_check_call_state(call, AFS_CALL_SV_REPLYING))
return afs_io_error(call, afs_io_error_cm_reply); return afs_io_error(call, afs_io_error_cm_reply);
return afs_find_cm_server_by_peer(call); return 0;
} }
/* /*
@ -667,9 +615,5 @@ static int afs_deliver_yfs_cb_callback(struct afs_call *call)
if (!afs_check_call_state(call, AFS_CALL_SV_REPLYING)) if (!afs_check_call_state(call, AFS_CALL_SV_REPLYING))
return afs_io_error(call, afs_io_error_cm_reply); return afs_io_error(call, afs_io_error_cm_reply);
return 0;
/* We'll need the file server record as that tells us which set of
* vnodes to operate upon.
*/
return afs_find_cm_server_by_peer(call);
} }

View File

@ -235,20 +235,20 @@ out:
* Probe all of a fileserver's addresses to find out the best route and to * Probe all of a fileserver's addresses to find out the best route and to
* query its capabilities. * query its capabilities.
*/ */
void afs_fs_probe_fileserver(struct afs_net *net, struct afs_server *server, int afs_fs_probe_fileserver(struct afs_net *net, struct afs_server *server,
struct afs_addr_list *new_alist, struct key *key) struct afs_addr_list *new_alist, struct key *key)
{ {
struct afs_endpoint_state *estate, *old; struct afs_endpoint_state *estate, *old;
struct afs_addr_list *alist; struct afs_addr_list *old_alist = NULL, *alist;
unsigned long unprobed; unsigned long unprobed;
_enter("%pU", &server->uuid); _enter("%pU", &server->uuid);
estate = kzalloc(sizeof(*estate), GFP_KERNEL); estate = kzalloc(sizeof(*estate), GFP_KERNEL);
if (!estate) if (!estate)
return; return -ENOMEM;
refcount_set(&estate->ref, 1); refcount_set(&estate->ref, 2);
estate->server_id = server->debug_id; estate->server_id = server->debug_id;
estate->rtt = UINT_MAX; estate->rtt = UINT_MAX;
@ -256,21 +256,31 @@ void afs_fs_probe_fileserver(struct afs_net *net, struct afs_server *server,
old = rcu_dereference_protected(server->endpoint_state, old = rcu_dereference_protected(server->endpoint_state,
lockdep_is_held(&server->fs_lock)); lockdep_is_held(&server->fs_lock));
estate->responsive_set = old->responsive_set; if (old) {
estate->addresses = afs_get_addrlist(new_alist ?: old->addresses, estate->responsive_set = old->responsive_set;
afs_alist_trace_get_estate); if (!new_alist)
new_alist = old->addresses;
}
if (old_alist != new_alist)
afs_set_peer_appdata(server, old_alist, new_alist);
estate->addresses = afs_get_addrlist(new_alist, afs_alist_trace_get_estate);
alist = estate->addresses; alist = estate->addresses;
estate->probe_seq = ++server->probe_counter; estate->probe_seq = ++server->probe_counter;
atomic_set(&estate->nr_probing, alist->nr_addrs); atomic_set(&estate->nr_probing, alist->nr_addrs);
if (new_alist)
server->addr_version = new_alist->version;
rcu_assign_pointer(server->endpoint_state, estate); rcu_assign_pointer(server->endpoint_state, estate);
set_bit(AFS_ESTATE_SUPERSEDED, &old->flags);
write_unlock(&server->fs_lock); write_unlock(&server->fs_lock);
if (old)
set_bit(AFS_ESTATE_SUPERSEDED, &old->flags);
trace_afs_estate(estate->server_id, estate->probe_seq, refcount_read(&estate->ref), trace_afs_estate(estate->server_id, estate->probe_seq, refcount_read(&estate->ref),
afs_estate_trace_alloc_probe); afs_estate_trace_alloc_probe);
afs_get_address_preferences(net, alist); afs_get_address_preferences(net, new_alist);
server->probed_at = jiffies; server->probed_at = jiffies;
unprobed = (1UL << alist->nr_addrs) - 1; unprobed = (1UL << alist->nr_addrs) - 1;
@ -293,6 +303,8 @@ void afs_fs_probe_fileserver(struct afs_net *net, struct afs_server *server,
} }
afs_put_endpoint_state(old, afs_estate_trace_put_probe); afs_put_endpoint_state(old, afs_estate_trace_put_probe);
afs_put_endpoint_state(estate, afs_estate_trace_put_probe);
return 0;
} }
/* /*

View File

@ -1010,6 +1010,9 @@ extern int afs_merge_fs_addr4(struct afs_net *net, struct afs_addr_list *addr,
__be32 xdr, u16 port); __be32 xdr, u16 port);
extern int afs_merge_fs_addr6(struct afs_net *net, struct afs_addr_list *addr, extern int afs_merge_fs_addr6(struct afs_net *net, struct afs_addr_list *addr,
__be32 *xdr, u16 port); __be32 *xdr, u16 port);
void afs_set_peer_appdata(struct afs_server *server,
struct afs_addr_list *old_alist,
struct afs_addr_list *new_alist);
/* /*
* addr_prefs.c * addr_prefs.c
@ -1207,8 +1210,8 @@ struct afs_endpoint_state *afs_get_endpoint_state(struct afs_endpoint_state *est
enum afs_estate_trace where); enum afs_estate_trace where);
void afs_put_endpoint_state(struct afs_endpoint_state *estate, enum afs_estate_trace where); void afs_put_endpoint_state(struct afs_endpoint_state *estate, enum afs_estate_trace where);
extern void afs_fileserver_probe_result(struct afs_call *); extern void afs_fileserver_probe_result(struct afs_call *);
void afs_fs_probe_fileserver(struct afs_net *net, struct afs_server *server, int afs_fs_probe_fileserver(struct afs_net *net, struct afs_server *server,
struct afs_addr_list *new_addrs, struct key *key); struct afs_addr_list *new_alist, struct key *key);
int afs_wait_for_fs_probes(struct afs_operation *op, struct afs_server_state *states, bool intr); int afs_wait_for_fs_probes(struct afs_operation *op, struct afs_server_state *states, bool intr);
extern void afs_probe_fileserver(struct afs_net *, struct afs_server *); extern void afs_probe_fileserver(struct afs_net *, struct afs_server *);
extern void afs_fs_probe_dispatcher(struct work_struct *); extern void afs_fs_probe_dispatcher(struct work_struct *);
@ -1509,7 +1512,7 @@ extern void __exit afs_clean_up_permit_cache(void);
*/ */
extern spinlock_t afs_server_peer_lock; extern spinlock_t afs_server_peer_lock;
extern struct afs_server *afs_find_server(struct afs_net *, const struct rxrpc_peer *); struct afs_server *afs_find_server(const struct rxrpc_peer *peer);
extern struct afs_server *afs_find_server_by_uuid(struct afs_net *, const uuid_t *); extern struct afs_server *afs_find_server_by_uuid(struct afs_net *, const uuid_t *);
extern struct afs_server *afs_lookup_server(struct afs_cell *, struct key *, const uuid_t *, u32); extern struct afs_server *afs_lookup_server(struct afs_cell *, struct key *, const uuid_t *, u32);
extern struct afs_server *afs_get_server(struct afs_server *, enum afs_server_trace); extern struct afs_server *afs_get_server(struct afs_server *, enum afs_server_trace);

View File

@ -444,8 +444,6 @@ static int afs_proc_servers_show(struct seq_file *m, void *v)
} }
server = list_entry(v, struct afs_server, proc_link); server = list_entry(v, struct afs_server, proc_link);
estate = rcu_dereference(server->endpoint_state);
alist = estate->addresses;
seq_printf(m, "%pU %3d %3d %s\n", seq_printf(m, "%pU %3d %3d %s\n",
&server->uuid, &server->uuid,
refcount_read(&server->ref), refcount_read(&server->ref),
@ -455,10 +453,16 @@ static int afs_proc_servers_show(struct seq_file *m, void *v)
server->flags, server->rtt); server->flags, server->rtt);
seq_printf(m, " - probe: last=%d\n", seq_printf(m, " - probe: last=%d\n",
(int)(jiffies - server->probed_at) / HZ); (int)(jiffies - server->probed_at) / HZ);
estate = rcu_dereference(server->endpoint_state);
if (!estate)
goto out;
failed = estate->failed_set; failed = estate->failed_set;
seq_printf(m, " - ESTATE pq=%x np=%u rsp=%lx f=%lx\n", seq_printf(m, " - ESTATE pq=%x np=%u rsp=%lx f=%lx\n",
estate->probe_seq, atomic_read(&estate->nr_probing), estate->probe_seq, atomic_read(&estate->nr_probing),
estate->responsive_set, estate->failed_set); estate->responsive_set, estate->failed_set);
alist = estate->addresses;
seq_printf(m, " - ALIST v=%u ap=%u\n", seq_printf(m, " - ALIST v=%u ap=%u\n",
alist->version, alist->addr_pref_version); alist->version, alist->addr_pref_version);
for (i = 0; i < alist->nr_addrs; i++) { for (i = 0; i < alist->nr_addrs; i++) {
@ -471,6 +475,8 @@ static int afs_proc_servers_show(struct seq_file *m, void *v)
rxrpc_kernel_get_srtt(addr->peer), rxrpc_kernel_get_srtt(addr->peer),
addr->last_error, addr->prio); addr->last_error, addr->prio);
} }
out:
return 0; return 0;
} }

View File

@ -766,8 +766,14 @@ static void afs_rx_discard_new_call(struct rxrpc_call *rxcall,
static void afs_rx_new_call(struct sock *sk, struct rxrpc_call *rxcall, static void afs_rx_new_call(struct sock *sk, struct rxrpc_call *rxcall,
unsigned long user_call_ID) unsigned long user_call_ID)
{ {
struct afs_call *call = (struct afs_call *)user_call_ID;
struct afs_net *net = afs_sock2net(sk); struct afs_net *net = afs_sock2net(sk);
call->peer = rxrpc_kernel_get_call_peer(sk->sk_socket, call->rxcall);
call->server = afs_find_server(call->peer);
if (!call->server)
trace_afs_cm_no_server(call, rxrpc_kernel_remote_srx(call->peer));
queue_work(afs_wq, &net->charge_preallocation_work); queue_work(afs_wq, &net->charge_preallocation_work);
} }

View File

@ -21,42 +21,13 @@ static void __afs_put_server(struct afs_net *, struct afs_server *);
/* /*
* Find a server by one of its addresses. * Find a server by one of its addresses.
*/ */
struct afs_server *afs_find_server(struct afs_net *net, const struct rxrpc_peer *peer) struct afs_server *afs_find_server(const struct rxrpc_peer *peer)
{ {
const struct afs_endpoint_state *estate; struct afs_server *server = (struct afs_server *)rxrpc_kernel_get_peer_data(peer);
const struct afs_addr_list *alist;
struct afs_server *server = NULL;
unsigned int i;
int seq = 1;
rcu_read_lock(); if (!server)
return NULL;
do { return afs_maybe_use_server(server, afs_server_trace_use_cm_call);
if (server)
afs_unuse_server_notime(net, server, afs_server_trace_unuse_find_rsq);
server = NULL;
seq++; /* 2 on the 1st/lockless path, otherwise odd */
read_seqbegin_or_lock(&net->fs_addr_lock, &seq);
hlist_for_each_entry_rcu(server, &net->fs_addresses, addr_link) {
estate = rcu_dereference(server->endpoint_state);
alist = estate->addresses;
for (i = 0; i < alist->nr_addrs; i++)
if (alist->addrs[i].peer == peer)
goto found;
}
server = NULL;
continue;
found:
server = afs_maybe_use_server(server, afs_server_trace_use_by_addr);
} while (need_seqretry(&net->fs_addr_lock, seq));
done_seqretry(&net->fs_addr_lock, seq);
rcu_read_unlock();
return server;
} }
/* /*
@ -468,9 +439,16 @@ static void afs_give_up_callbacks(struct afs_net *net, struct afs_server *server
*/ */
static void afs_destroy_server(struct afs_net *net, struct afs_server *server) static void afs_destroy_server(struct afs_net *net, struct afs_server *server)
{ {
struct afs_endpoint_state *estate;
if (test_bit(AFS_SERVER_FL_MAY_HAVE_CB, &server->flags)) if (test_bit(AFS_SERVER_FL_MAY_HAVE_CB, &server->flags))
afs_give_up_callbacks(net, server); afs_give_up_callbacks(net, server);
/* Unbind the rxrpc_peer records from the server. */
estate = rcu_access_pointer(server->endpoint_state);
if (estate)
afs_set_peer_appdata(server, estate->addresses, NULL);
afs_put_server(net, server, afs_server_trace_destroy); afs_put_server(net, server, afs_server_trace_destroy);
} }

View File

@ -140,12 +140,10 @@ enum yfs_cm_operation {
EM(afs_server_trace_see_expired, "SEE expd ") \ EM(afs_server_trace_see_expired, "SEE expd ") \
EM(afs_server_trace_unuse_call, "UNU call ") \ EM(afs_server_trace_unuse_call, "UNU call ") \
EM(afs_server_trace_unuse_create_fail, "UNU cfail") \ EM(afs_server_trace_unuse_create_fail, "UNU cfail") \
EM(afs_server_trace_unuse_find_rsq, "UNU f-rsq") \
EM(afs_server_trace_unuse_slist, "UNU slist") \ EM(afs_server_trace_unuse_slist, "UNU slist") \
EM(afs_server_trace_unuse_slist_isort, "UNU isort") \ EM(afs_server_trace_unuse_slist_isort, "UNU isort") \
EM(afs_server_trace_unuse_uuid_rsq, "PUT u-req") \ EM(afs_server_trace_unuse_uuid_rsq, "PUT u-req") \
EM(afs_server_trace_update, "UPDATE ") \ EM(afs_server_trace_update, "UPDATE ") \
EM(afs_server_trace_use_by_addr, "USE addr ") \
EM(afs_server_trace_use_by_uuid, "USE uuid ") \ EM(afs_server_trace_use_by_uuid, "USE uuid ") \
EM(afs_server_trace_use_cm_call, "USE cm-cl") \ EM(afs_server_trace_use_cm_call, "USE cm-cl") \
EM(afs_server_trace_use_get_caps, "USE gcaps") \ EM(afs_server_trace_use_get_caps, "USE gcaps") \
@ -1281,7 +1279,7 @@ TRACE_EVENT(afs_bulkstat_error,
); );
TRACE_EVENT(afs_cm_no_server, TRACE_EVENT(afs_cm_no_server,
TP_PROTO(struct afs_call *call, struct sockaddr_rxrpc *srx), TP_PROTO(struct afs_call *call, const struct sockaddr_rxrpc *srx),
TP_ARGS(call, srx), TP_ARGS(call, srx),

View File

@ -461,7 +461,7 @@ void rxrpc_destroy_all_peers(struct rxrpc_net *rxnet)
continue; continue;
hlist_for_each_entry(peer, &rxnet->peer_hash[i], hash_link) { hlist_for_each_entry(peer, &rxnet->peer_hash[i], hash_link) {
pr_err("Leaked peer %u {%u} %pISp\n", pr_err("Leaked peer %x {%u} %pISp\n",
peer->debug_id, peer->debug_id,
refcount_read(&peer->ref), refcount_read(&peer->ref),
&peer->srx.transport); &peer->srx.transport);
@ -478,7 +478,7 @@ void rxrpc_destroy_all_peers(struct rxrpc_net *rxnet)
*/ */
struct rxrpc_peer *rxrpc_kernel_get_call_peer(struct socket *sock, struct rxrpc_call *call) struct rxrpc_peer *rxrpc_kernel_get_call_peer(struct socket *sock, struct rxrpc_call *call)
{ {
return call->peer; return rxrpc_get_peer(call->peer, rxrpc_peer_get_application);
} }
EXPORT_SYMBOL(rxrpc_kernel_get_call_peer); EXPORT_SYMBOL(rxrpc_kernel_get_call_peer);