Merge branch 'rxrpc-miscellaneous-fixes'
David Howells says: ==================== rxrpc: Miscellaneous fixes Here some miscellaneous fixes for AF_RXRPC: (1) Fix a race in the I/O thread vs UDP socket setup. (2) Fix an uninitialised variable. ==================== Link: https://patch.msgid.link/20241001132702.3122709-1-dhowells@redhat.com Signed-off-by: Jakub Kicinski <kuba@kernel.org>
This commit is contained in:
commit
35f1210879
@ -1056,7 +1056,7 @@ bool rxrpc_direct_abort(struct sk_buff *skb, enum rxrpc_abort_reason why,
|
|||||||
int rxrpc_io_thread(void *data);
|
int rxrpc_io_thread(void *data);
|
||||||
static inline void rxrpc_wake_up_io_thread(struct rxrpc_local *local)
|
static inline void rxrpc_wake_up_io_thread(struct rxrpc_local *local)
|
||||||
{
|
{
|
||||||
wake_up_process(local->io_thread);
|
wake_up_process(READ_ONCE(local->io_thread));
|
||||||
}
|
}
|
||||||
|
|
||||||
static inline bool rxrpc_protocol_error(struct sk_buff *skb, enum rxrpc_abort_reason why)
|
static inline bool rxrpc_protocol_error(struct sk_buff *skb, enum rxrpc_abort_reason why)
|
||||||
|
@ -27,11 +27,17 @@ int rxrpc_encap_rcv(struct sock *udp_sk, struct sk_buff *skb)
|
|||||||
{
|
{
|
||||||
struct sk_buff_head *rx_queue;
|
struct sk_buff_head *rx_queue;
|
||||||
struct rxrpc_local *local = rcu_dereference_sk_user_data(udp_sk);
|
struct rxrpc_local *local = rcu_dereference_sk_user_data(udp_sk);
|
||||||
|
struct task_struct *io_thread;
|
||||||
|
|
||||||
if (unlikely(!local)) {
|
if (unlikely(!local)) {
|
||||||
kfree_skb(skb);
|
kfree_skb(skb);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
io_thread = READ_ONCE(local->io_thread);
|
||||||
|
if (!io_thread) {
|
||||||
|
kfree_skb(skb);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
if (skb->tstamp == 0)
|
if (skb->tstamp == 0)
|
||||||
skb->tstamp = ktime_get_real();
|
skb->tstamp = ktime_get_real();
|
||||||
|
|
||||||
@ -47,7 +53,7 @@ int rxrpc_encap_rcv(struct sock *udp_sk, struct sk_buff *skb)
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
skb_queue_tail(rx_queue, skb);
|
skb_queue_tail(rx_queue, skb);
|
||||||
rxrpc_wake_up_io_thread(local);
|
wake_up_process(io_thread);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -565,7 +571,7 @@ int rxrpc_io_thread(void *data)
|
|||||||
__set_current_state(TASK_RUNNING);
|
__set_current_state(TASK_RUNNING);
|
||||||
rxrpc_see_local(local, rxrpc_local_stop);
|
rxrpc_see_local(local, rxrpc_local_stop);
|
||||||
rxrpc_destroy_local(local);
|
rxrpc_destroy_local(local);
|
||||||
local->io_thread = NULL;
|
WRITE_ONCE(local->io_thread, NULL);
|
||||||
rxrpc_see_local(local, rxrpc_local_stopped);
|
rxrpc_see_local(local, rxrpc_local_stopped);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@ -232,7 +232,7 @@ static int rxrpc_open_socket(struct rxrpc_local *local, struct net *net)
|
|||||||
}
|
}
|
||||||
|
|
||||||
wait_for_completion(&local->io_thread_ready);
|
wait_for_completion(&local->io_thread_ready);
|
||||||
local->io_thread = io_thread;
|
WRITE_ONCE(local->io_thread, io_thread);
|
||||||
_leave(" = 0");
|
_leave(" = 0");
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
|
@ -303,6 +303,11 @@ static int rxrpc_send_data(struct rxrpc_sock *rx,
|
|||||||
sk_clear_bit(SOCKWQ_ASYNC_NOSPACE, sk);
|
sk_clear_bit(SOCKWQ_ASYNC_NOSPACE, sk);
|
||||||
|
|
||||||
reload:
|
reload:
|
||||||
|
txb = call->tx_pending;
|
||||||
|
call->tx_pending = NULL;
|
||||||
|
if (txb)
|
||||||
|
rxrpc_see_txbuf(txb, rxrpc_txbuf_see_send_more);
|
||||||
|
|
||||||
ret = -EPIPE;
|
ret = -EPIPE;
|
||||||
if (sk->sk_shutdown & SEND_SHUTDOWN)
|
if (sk->sk_shutdown & SEND_SHUTDOWN)
|
||||||
goto maybe_error;
|
goto maybe_error;
|
||||||
@ -329,11 +334,6 @@ reload:
|
|||||||
goto maybe_error;
|
goto maybe_error;
|
||||||
}
|
}
|
||||||
|
|
||||||
txb = call->tx_pending;
|
|
||||||
call->tx_pending = NULL;
|
|
||||||
if (txb)
|
|
||||||
rxrpc_see_txbuf(txb, rxrpc_txbuf_see_send_more);
|
|
||||||
|
|
||||||
do {
|
do {
|
||||||
if (!txb) {
|
if (!txb) {
|
||||||
size_t remain;
|
size_t remain;
|
||||||
|
Loading…
Reference in New Issue
Block a user