1 // SPDX-License-Identifier: GPL-2.0-or-later
2 /* RxRPC Tx data buffering.
4 * Copyright (C) 2022 Red Hat, Inc. All Rights Reserved.
8 #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
10 #include <linux/slab.h>
11 #include "ar-internal.h"
13 static atomic_t rxrpc_txbuf_debug_ids;
14 atomic_t rxrpc_nr_txbuf;
17 * Allocate and partially initialise a data transmission buffer.
19 struct rxrpc_txbuf *rxrpc_alloc_data_txbuf(struct rxrpc_call *call, size_t data_size,
20 size_t data_align, gfp_t gfp)
22 struct rxrpc_txbuf *txb;
23 size_t total, doff, jsize = sizeof(struct rxrpc_jumbo_header);
26 txb = kzalloc(sizeof(*txb), gfp);
30 /* We put a jumbo header in the buffer, but not a full wire header to
31 * avoid delayed-corruption problems with zerocopy.
33 doff = round_up(jsize, data_align);
34 total = doff + data_size;
36 data_align = umax(data_align, L1_CACHE_BYTES);
37 mutex_lock(&call->conn->tx_data_alloc_lock);
38 buf = page_frag_alloc_align(&call->conn->tx_data_alloc, total, gfp,
40 mutex_unlock(&call->conn->tx_data_alloc_lock);
46 refcount_set(&txb->ref, 1);
47 txb->call_debug_id = call->debug_id;
48 txb->debug_id = atomic_inc_return(&rxrpc_txbuf_debug_ids);
49 txb->alloc_size = data_size;
50 txb->space = data_size;
52 txb->flags = call->conn->out_clientflag;
53 txb->seq = call->send_top + 1;
54 txb->data = buf + doff;
56 trace_rxrpc_txbuf(txb->debug_id, txb->call_debug_id, txb->seq, 1,
57 rxrpc_txbuf_alloc_data);
59 atomic_inc(&rxrpc_nr_txbuf);
63 void rxrpc_get_txbuf(struct rxrpc_txbuf *txb, enum rxrpc_txbuf_trace what)
67 __refcount_inc(&txb->ref, &r);
68 trace_rxrpc_txbuf(txb->debug_id, txb->call_debug_id, txb->seq, r + 1, what);
71 void rxrpc_see_txbuf(struct rxrpc_txbuf *txb, enum rxrpc_txbuf_trace what)
73 int r = refcount_read(&txb->ref);
75 trace_rxrpc_txbuf(txb->debug_id, txb->call_debug_id, txb->seq, r, what);
78 static void rxrpc_free_txbuf(struct rxrpc_txbuf *txb)
80 trace_rxrpc_txbuf(txb->debug_id, txb->call_debug_id, txb->seq, 0,
83 page_frag_free(txb->data);
85 atomic_dec(&rxrpc_nr_txbuf);
88 void rxrpc_put_txbuf(struct rxrpc_txbuf *txb, enum rxrpc_txbuf_trace what)
90 unsigned int debug_id, call_debug_id;
96 debug_id = txb->debug_id;
97 call_debug_id = txb->call_debug_id;
99 dead = __refcount_dec_and_test(&txb->ref, &r);
100 trace_rxrpc_txbuf(debug_id, call_debug_id, seq, r - 1, what);
102 rxrpc_free_txbuf(txb);