Commit 6b47808f authored by Jakub Kicinski's avatar Jakub Kicinski Committed by David S. Miller

net: tls: avoid discarding data on record close

TLS records end with a 16B tag. For TLS device offload we only
need to make space for this tag in the stream, the device will
generate and replace it with the actual calculated tag.

Long time ago the code would just re-reference the head frag
which mostly worked but was suboptimal because it prevented TCP
from combining the record into a single skb frag. I'm not sure
if it was correct as the first frag may be shorter than the tag.

The commit under fixes tried to replace that with using the page
frag and if the allocation failed rolling back the data, if record
was long enough. It achieves better fragment coalescing but is
also buggy.

We don't roll back the iterator, so unless we're at the end of
send we'll skip the data we designated as tag and start the
next record as if the rollback never happened.
There's also the possibility that the record was constructed
with MSG_MORE and the data came from a different syscall and
we already told the user space that we "got it".

Allocate a single dummy page and use it as fallback.

Found by code inspection, and proven by forcing allocation
failures.

Fixes: e7b159a4 ("net/tls: remove the record tail optimization")
Signed-off-by: default avatarJakub Kicinski <kuba@kernel.org>
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parent a47e598f
...@@ -52,6 +52,8 @@ static LIST_HEAD(tls_device_list); ...@@ -52,6 +52,8 @@ static LIST_HEAD(tls_device_list);
static LIST_HEAD(tls_device_down_list); static LIST_HEAD(tls_device_down_list);
static DEFINE_SPINLOCK(tls_device_lock); static DEFINE_SPINLOCK(tls_device_lock);
static struct page *dummy_page;
static void tls_device_free_ctx(struct tls_context *ctx) static void tls_device_free_ctx(struct tls_context *ctx)
{ {
if (ctx->tx_conf == TLS_HW) { if (ctx->tx_conf == TLS_HW) {
...@@ -312,36 +314,33 @@ static int tls_push_record(struct sock *sk, ...@@ -312,36 +314,33 @@ static int tls_push_record(struct sock *sk,
return tls_push_sg(sk, ctx, offload_ctx->sg_tx_data, 0, flags); return tls_push_sg(sk, ctx, offload_ctx->sg_tx_data, 0, flags);
} }
static int tls_device_record_close(struct sock *sk, static void tls_device_record_close(struct sock *sk,
struct tls_context *ctx, struct tls_context *ctx,
struct tls_record_info *record, struct tls_record_info *record,
struct page_frag *pfrag, struct page_frag *pfrag,
unsigned char record_type) unsigned char record_type)
{ {
struct tls_prot_info *prot = &ctx->prot_info; struct tls_prot_info *prot = &ctx->prot_info;
int ret; struct page_frag dummy_tag_frag;
/* append tag /* append tag
* device will fill in the tag, we just need to append a placeholder * device will fill in the tag, we just need to append a placeholder
* use socket memory to improve coalescing (re-using a single buffer * use socket memory to improve coalescing (re-using a single buffer
* increases frag count) * increases frag count)
* if we can't allocate memory now, steal some back from data * if we can't allocate memory now use the dummy page
*/ */
if (likely(skb_page_frag_refill(prot->tag_size, pfrag, if (unlikely(pfrag->size - pfrag->offset < prot->tag_size) &&
sk->sk_allocation))) { !skb_page_frag_refill(prot->tag_size, pfrag, sk->sk_allocation)) {
ret = 0; dummy_tag_frag.page = dummy_page;
tls_append_frag(record, pfrag, prot->tag_size); dummy_tag_frag.offset = 0;
} else { pfrag = &dummy_tag_frag;
ret = prot->tag_size;
if (record->len <= prot->overhead_size)
return -ENOMEM;
} }
tls_append_frag(record, pfrag, prot->tag_size);
/* fill prepend */ /* fill prepend */
tls_fill_prepend(ctx, skb_frag_address(&record->frags[0]), tls_fill_prepend(ctx, skb_frag_address(&record->frags[0]),
record->len - prot->overhead_size, record->len - prot->overhead_size,
record_type); record_type);
return ret;
} }
static int tls_create_new_record(struct tls_offload_context_tx *offload_ctx, static int tls_create_new_record(struct tls_offload_context_tx *offload_ctx,
...@@ -541,18 +540,8 @@ static int tls_push_data(struct sock *sk, ...@@ -541,18 +540,8 @@ static int tls_push_data(struct sock *sk,
if (done || record->len >= max_open_record_len || if (done || record->len >= max_open_record_len ||
(record->num_frags >= MAX_SKB_FRAGS - 1)) { (record->num_frags >= MAX_SKB_FRAGS - 1)) {
rc = tls_device_record_close(sk, tls_ctx, record, tls_device_record_close(sk, tls_ctx, record,
pfrag, record_type); pfrag, record_type);
if (rc) {
if (rc > 0) {
size += rc;
} else {
size = orig_size;
destroy_record(record);
ctx->open_record = NULL;
break;
}
}
rc = tls_push_record(sk, rc = tls_push_record(sk,
tls_ctx, tls_ctx,
...@@ -1450,14 +1439,26 @@ int __init tls_device_init(void) ...@@ -1450,14 +1439,26 @@ int __init tls_device_init(void)
{ {
int err; int err;
destruct_wq = alloc_workqueue("ktls_device_destruct", 0, 0); dummy_page = alloc_page(GFP_KERNEL);
if (!destruct_wq) if (!dummy_page)
return -ENOMEM; return -ENOMEM;
destruct_wq = alloc_workqueue("ktls_device_destruct", 0, 0);
if (!destruct_wq) {
err = -ENOMEM;
goto err_free_dummy;
}
err = register_netdevice_notifier(&tls_dev_notifier); err = register_netdevice_notifier(&tls_dev_notifier);
if (err) if (err)
destroy_workqueue(destruct_wq); goto err_destroy_wq;
return 0;
err_destroy_wq:
destroy_workqueue(destruct_wq);
err_free_dummy:
put_page(dummy_page);
return err; return err;
} }
...@@ -1466,4 +1467,5 @@ void __exit tls_device_cleanup(void) ...@@ -1466,4 +1467,5 @@ void __exit tls_device_cleanup(void)
unregister_netdevice_notifier(&tls_dev_notifier); unregister_netdevice_notifier(&tls_dev_notifier);
destroy_workqueue(destruct_wq); destroy_workqueue(destruct_wq);
clean_acked_data_flush(); clean_acked_data_flush();
put_page(dummy_page);
} }
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment