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/ # v4pull/1113/merge
parent
f3a123b254
commit
40e8b52fe8
|
|
@ -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);
|
||||||
|
}
|
||||||
|
|
|
||||||
|
|
@ -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);
|
|
||||||
}
|
}
|
||||||
|
|
|
||||||
|
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
|
|
||||||
|
|
@ -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);
|
||||||
|
|
|
||||||
|
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -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);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -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);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -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),
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -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);
|
||||||
|
|
||||||
|
|
|
||||||
Loading…
Reference in New Issue