mirror of
https://github.com/torvalds/linux.git
synced 2024-11-11 06:31:49 +00:00
rxrpc: Fix lack of conn cleanup when local endpoint is cleaned up [ver #2]
When a local endpoint is ceases to be in use, such as when the kafs module
is unloaded, the kernel will emit an assertion failure if there are any
outstanding client connections:
rxrpc: Assertion failed
------------[ cut here ]------------
kernel BUG at net/rxrpc/local_object.c:433!
and even beyond that, will evince other oopses if there are service
connections still present.
Fix this by:
(1) Removing the triggering of connection reaping when an rxrpc socket is
released. These don't actually clean up the connections anyway - and
further, the local endpoint may still be in use through another
socket.
(2) Mark the local endpoint as dead when we start the process of tearing
it down.
(3) When destroying a local endpoint, strip all of its client connections
from the idle list and discard the ref on each that the list was
holding.
(4) When destroying a local endpoint, call the service connection reaper
directly (rather than through a workqueue) to immediately kill off all
outstanding service connections.
(5) Make the service connection reaper reap connections for which the
local endpoint is marked dead.
Only after destroying the connections can we close the socket lest we get
an oops in a workqueue that's looking at a connection or a peer.
Fixes: 3d18cbb7fd
("rxrpc: Fix conn expiry timers")
Signed-off-by: David Howells <dhowells@redhat.com>
Tested-by: Marc Dionne <marc.dionne@auristor.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
parent
a285c1fa39
commit
d12040b693
@ -862,7 +862,6 @@ static void rxrpc_sock_destructor(struct sock *sk)
|
||||
static int rxrpc_release_sock(struct sock *sk)
|
||||
{
|
||||
struct rxrpc_sock *rx = rxrpc_sk(sk);
|
||||
struct rxrpc_net *rxnet = rxrpc_net(sock_net(&rx->sk));
|
||||
|
||||
_enter("%p{%d,%d}", sk, sk->sk_state, refcount_read(&sk->sk_refcnt));
|
||||
|
||||
@ -898,8 +897,6 @@ static int rxrpc_release_sock(struct sock *sk)
|
||||
rxrpc_release_calls_on_socket(rx);
|
||||
flush_workqueue(rxrpc_workqueue);
|
||||
rxrpc_purge_queue(&sk->sk_receive_queue);
|
||||
rxrpc_queue_work(&rxnet->service_conn_reaper);
|
||||
rxrpc_queue_work(&rxnet->client_conn_reaper);
|
||||
|
||||
rxrpc_unuse_local(rx->local);
|
||||
rx->local = NULL;
|
||||
|
@ -910,6 +910,7 @@ void rxrpc_disconnect_client_call(struct rxrpc_call *);
|
||||
void rxrpc_put_client_conn(struct rxrpc_connection *);
|
||||
void rxrpc_discard_expired_client_conns(struct work_struct *);
|
||||
void rxrpc_destroy_all_client_connections(struct rxrpc_net *);
|
||||
void rxrpc_clean_up_local_conns(struct rxrpc_local *);
|
||||
|
||||
/*
|
||||
* conn_event.c
|
||||
|
@ -1162,3 +1162,47 @@ void rxrpc_destroy_all_client_connections(struct rxrpc_net *rxnet)
|
||||
|
||||
_leave("");
|
||||
}
|
||||
|
||||
/*
|
||||
* Clean up the client connections on a local endpoint.
|
||||
*/
|
||||
void rxrpc_clean_up_local_conns(struct rxrpc_local *local)
|
||||
{
|
||||
struct rxrpc_connection *conn, *tmp;
|
||||
struct rxrpc_net *rxnet = local->rxnet;
|
||||
unsigned int nr_active;
|
||||
LIST_HEAD(graveyard);
|
||||
|
||||
_enter("");
|
||||
|
||||
spin_lock(&rxnet->client_conn_cache_lock);
|
||||
nr_active = rxnet->nr_active_client_conns;
|
||||
|
||||
list_for_each_entry_safe(conn, tmp, &rxnet->idle_client_conns,
|
||||
cache_link) {
|
||||
if (conn->params.local == local) {
|
||||
ASSERTCMP(conn->cache_state, ==, RXRPC_CONN_CLIENT_IDLE);
|
||||
|
||||
trace_rxrpc_client(conn, -1, rxrpc_client_discard);
|
||||
if (!test_and_clear_bit(RXRPC_CONN_EXPOSED, &conn->flags))
|
||||
BUG();
|
||||
conn->cache_state = RXRPC_CONN_CLIENT_INACTIVE;
|
||||
list_move(&conn->cache_link, &graveyard);
|
||||
nr_active--;
|
||||
}
|
||||
}
|
||||
|
||||
rxnet->nr_active_client_conns = nr_active;
|
||||
spin_unlock(&rxnet->client_conn_cache_lock);
|
||||
ASSERTCMP(nr_active, >=, 0);
|
||||
|
||||
while (!list_empty(&graveyard)) {
|
||||
conn = list_entry(graveyard.next,
|
||||
struct rxrpc_connection, cache_link);
|
||||
list_del_init(&conn->cache_link);
|
||||
|
||||
rxrpc_put_connection(conn);
|
||||
}
|
||||
|
||||
_leave(" [culled]");
|
||||
}
|
||||
|
@ -398,7 +398,7 @@ void rxrpc_service_connection_reaper(struct work_struct *work)
|
||||
if (conn->state == RXRPC_CONN_SERVICE_PREALLOC)
|
||||
continue;
|
||||
|
||||
if (rxnet->live) {
|
||||
if (rxnet->live && !conn->params.local->dead) {
|
||||
idle_timestamp = READ_ONCE(conn->idle_timestamp);
|
||||
expire_at = idle_timestamp + rxrpc_connection_expiry * HZ;
|
||||
if (conn->params.local->service_closed)
|
||||
|
@ -426,11 +426,14 @@ static void rxrpc_local_destroyer(struct rxrpc_local *local)
|
||||
|
||||
_enter("%d", local->debug_id);
|
||||
|
||||
local->dead = true;
|
||||
|
||||
mutex_lock(&rxnet->local_mutex);
|
||||
list_del_init(&local->link);
|
||||
mutex_unlock(&rxnet->local_mutex);
|
||||
|
||||
ASSERT(RB_EMPTY_ROOT(&local->client_conns));
|
||||
rxrpc_clean_up_local_conns(local);
|
||||
rxrpc_service_connection_reaper(&rxnet->service_conn_reaper);
|
||||
ASSERT(!local->service);
|
||||
|
||||
if (socket) {
|
||||
|
Loading…
Reference in New Issue
Block a user