2022-07-12 21:52:38 +01:00
# include <linux/kernel.h>
# include <linux/errno.h>
# include <linux/file.h>
# include <linux/slab.h>
# include <linux/net.h>
# include <linux/io_uring.h>
# include "io_uring.h"
# include "notif.h"
2022-07-12 21:52:41 +01:00
# include "rsrc.h"
2022-07-12 21:52:38 +01:00
2022-11-04 10:59:46 +00:00
static void io_notif_complete_tw_ext ( struct io_kiocb * notif , bool * locked )
2022-07-12 21:52:38 +01:00
{
2022-07-27 10:30:41 +01:00
struct io_notif_data * nd = io_notif_to_data ( notif ) ;
2022-07-12 21:52:38 +01:00
struct io_ring_ctx * ctx = notif - > ctx ;
2022-11-04 10:59:46 +00:00
if ( nd - > zc_report & & ( nd - > zc_copied | | ! nd - > zc_used ) )
notif - > cqe . res | = IORING_NOTIF_USAGE_ZC_COPIED ;
2022-07-27 10:30:41 +01:00
if ( nd - > account_pages & & ctx - > user ) {
__io_unaccount_mem ( ctx - > user , nd - > account_pages ) ;
nd - > account_pages = 0 ;
2022-07-12 21:52:44 +01:00
}
2022-11-04 10:59:45 +00:00
io_req_task_complete ( notif , locked ) ;
}
2022-11-04 10:59:43 +00:00
static void io_tx_ubuf_callback ( struct sk_buff * skb , struct ubuf_info * uarg ,
bool success )
2022-07-12 21:52:38 +01:00
{
2022-07-27 10:30:41 +01:00
struct io_notif_data * nd = container_of ( uarg , struct io_notif_data , uarg ) ;
struct io_kiocb * notif = cmd_to_io_kiocb ( nd ) ;
2022-07-12 21:52:39 +01:00
2022-11-04 10:59:45 +00:00
if ( refcount_dec_and_test ( & uarg - > refcnt ) )
io_req_task_work_add ( notif ) ;
}
static void io_tx_ubuf_callback_ext ( struct sk_buff * skb , struct ubuf_info * uarg ,
bool success )
{
struct io_notif_data * nd = container_of ( uarg , struct io_notif_data , uarg ) ;
2022-10-27 20:34:45 +02:00
if ( nd - > zc_report ) {
if ( success & & ! nd - > zc_used & & skb )
WRITE_ONCE ( nd - > zc_used , true ) ;
else if ( ! success & & ! nd - > zc_copied )
WRITE_ONCE ( nd - > zc_copied , true ) ;
}
2022-11-04 10:59:45 +00:00
io_tx_ubuf_callback ( skb , uarg , success ) ;
}
2022-10-27 20:34:45 +02:00
2022-11-04 10:59:45 +00:00
void io_notif_set_extended ( struct io_kiocb * notif )
{
struct io_notif_data * nd = io_notif_to_data ( notif ) ;
2022-11-04 10:59:46 +00:00
if ( nd - > uarg . callback ! = io_tx_ubuf_callback_ext ) {
nd - > account_pages = 0 ;
nd - > zc_report = false ;
nd - > zc_used = false ;
nd - > zc_copied = false ;
nd - > uarg . callback = io_tx_ubuf_callback_ext ;
notif - > io_task_work . func = io_notif_complete_tw_ext ;
}
2022-07-12 21:52:39 +01:00
}
2022-09-01 11:54:04 +01:00
struct io_kiocb * io_alloc_notif ( struct io_ring_ctx * ctx )
2022-07-12 21:52:38 +01:00
__must_hold ( & ctx - > uring_lock )
{
2022-07-27 10:30:41 +01:00
struct io_kiocb * notif ;
struct io_notif_data * nd ;
2023-01-23 14:37:16 +00:00
if ( unlikely ( ! io_alloc_req ( ctx , & notif ) ) )
2022-07-27 10:30:41 +01:00
return NULL ;
notif - > opcode = IORING_OP_NOP ;
notif - > flags = 0 ;
notif - > file = NULL ;
notif - > task = current ;
io_get_task_refs ( 1 ) ;
notif - > rsrc_node = NULL ;
2022-11-04 10:59:46 +00:00
notif - > io_task_work . func = io_req_task_complete ;
2022-07-27 10:30:41 +01:00
nd = io_notif_to_data ( notif ) ;
nd - > uarg . flags = SKBFL_ZEROCOPY_FRAG | SKBFL_DONT_ORPHAN ;
2022-11-04 10:59:43 +00:00
nd - > uarg . callback = io_tx_ubuf_callback ;
2022-07-27 10:30:41 +01:00
refcount_set ( & nd - > uarg . refcnt , 1 ) ;
2022-07-12 21:52:38 +01:00
return notif ;
}