2018-04-30 10:16:16 +03:00
/* Copyright (c) 2018, Mellanox Technologies All rights reserved.
*
* This software is available to you under a choice of one of two
* licenses . You may choose to be licensed under the terms of the GNU
* General Public License ( GPL ) Version 2 , available from the file
* COPYING in the main directory of this source tree , or the
* OpenIB . org BSD license below :
*
* Redistribution and use in source and binary forms , with or
* without modification , are permitted provided that the following
* conditions are met :
*
* - Redistributions of source code must retain the above
* copyright notice , this list of conditions and the following
* disclaimer .
*
* - Redistributions in binary form must reproduce the above
* copyright notice , this list of conditions and the following
* disclaimer in the documentation and / or other materials
* provided with the distribution .
*
* THE SOFTWARE IS PROVIDED " AS IS " , WITHOUT WARRANTY OF ANY KIND ,
* EXPRESS OR IMPLIED , INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
* MERCHANTABILITY , FITNESS FOR A PARTICULAR PURPOSE AND
* NONINFRINGEMENT . IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
* BE LIABLE FOR ANY CLAIM , DAMAGES OR OTHER LIABILITY , WHETHER IN AN
* ACTION OF CONTRACT , TORT OR OTHERWISE , ARISING FROM , OUT OF OR IN
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
* SOFTWARE .
*/
# include <crypto/aead.h>
# include <linux/highmem.h>
# include <linux/module.h>
# include <linux/netdevice.h>
# include <net/dst.h>
# include <net/inet_connection_sock.h>
# include <net/tcp.h>
# include <net/tls.h>
2019-10-04 16:19:22 -07:00
# include "trace.h"
2018-04-30 10:16:16 +03:00
/* device_offload_lock is used to synchronize tls_dev_add
* against NETDEV_DOWN notifications .
*/
static DECLARE_RWSEM ( device_offload_lock ) ;
static void tls_device_gc_task ( struct work_struct * work ) ;
static DECLARE_WORK ( tls_device_gc_work , tls_device_gc_task ) ;
static LIST_HEAD ( tls_device_gc_list ) ;
static LIST_HEAD ( tls_device_list ) ;
static DEFINE_SPINLOCK ( tls_device_lock ) ;
static void tls_device_free_ctx ( struct tls_context * ctx )
{
2019-04-10 11:04:30 -07:00
if ( ctx - > tx_conf = = TLS_HW ) {
2018-07-13 14:33:43 +03:00
kfree ( tls_offload_ctx_tx ( ctx ) ) ;
2019-04-10 11:04:30 -07:00
kfree ( ctx - > tx . rec_seq ) ;
kfree ( ctx - > tx . iv ) ;
}
2018-07-13 14:33:43 +03:00
if ( ctx - > rx_conf = = TLS_HW )
kfree ( tls_offload_ctx_rx ( ctx ) ) ;
2018-04-30 10:16:16 +03:00
2019-08-30 12:25:47 +02:00
tls_ctx_free ( NULL , ctx ) ;
2018-04-30 10:16:16 +03:00
}
static void tls_device_gc_task ( struct work_struct * work )
{
struct tls_context * ctx , * tmp ;
unsigned long flags ;
LIST_HEAD ( gc_list ) ;
spin_lock_irqsave ( & tls_device_lock , flags ) ;
list_splice_init ( & tls_device_gc_list , & gc_list ) ;
spin_unlock_irqrestore ( & tls_device_lock , flags ) ;
list_for_each_entry_safe ( ctx , tmp , & gc_list , list ) {
struct net_device * netdev = ctx - > netdev ;
2018-07-13 14:33:43 +03:00
if ( netdev & & ctx - > tx_conf = = TLS_HW ) {
2018-04-30 10:16:16 +03:00
netdev - > tlsdev_ops - > tls_dev_del ( netdev , ctx ,
TLS_OFFLOAD_CTX_DIR_TX ) ;
dev_put ( netdev ) ;
2018-07-13 14:33:43 +03:00
ctx - > netdev = NULL ;
2018-04-30 10:16:16 +03:00
}
list_del ( & ctx - > list ) ;
tls_device_free_ctx ( ctx ) ;
}
}
static void tls_device_queue_ctx_destruction ( struct tls_context * ctx )
{
unsigned long flags ;
spin_lock_irqsave ( & tls_device_lock , flags ) ;
list_move_tail ( & ctx - > list , & tls_device_gc_list ) ;
/* schedule_work inside the spinlock
* to make sure tls_device_down waits for that work .
*/
schedule_work ( & tls_device_gc_work ) ;
spin_unlock_irqrestore ( & tls_device_lock , flags ) ;
}
/* We assume that the socket is already connected */
static struct net_device * get_netdev_for_sock ( struct sock * sk )
{
struct dst_entry * dst = sk_dst_get ( sk ) ;
struct net_device * netdev = NULL ;
if ( likely ( dst ) ) {
netdev = dst - > dev ;
dev_hold ( netdev ) ;
}
dst_release ( dst ) ;
return netdev ;
}
static void destroy_record ( struct tls_record_info * record )
{
2019-09-06 22:29:57 -07:00
int i ;
2018-04-30 10:16:16 +03:00
2019-09-06 22:29:57 -07:00
for ( i = 0 ; i < record - > num_frags ; i + + )
__skb_frag_unref ( & record - > frags [ i ] ) ;
2018-04-30 10:16:16 +03:00
kfree ( record ) ;
}
2018-07-13 14:33:39 +03:00
static void delete_all_records ( struct tls_offload_context_tx * offload_ctx )
2018-04-30 10:16:16 +03:00
{
struct tls_record_info * info , * temp ;
list_for_each_entry_safe ( info , temp , & offload_ctx - > records_list , list ) {
list_del ( & info - > list ) ;
destroy_record ( info ) ;
}
offload_ctx - > retransmit_hint = NULL ;
}
static void tls_icsk_clean_acked ( struct sock * sk , u32 acked_seq )
{
struct tls_context * tls_ctx = tls_get_ctx ( sk ) ;
struct tls_record_info * info , * temp ;
2018-07-13 14:33:39 +03:00
struct tls_offload_context_tx * ctx ;
2018-04-30 10:16:16 +03:00
u64 deleted_records = 0 ;
unsigned long flags ;
if ( ! tls_ctx )
return ;
2018-07-13 14:33:39 +03:00
ctx = tls_offload_ctx_tx ( tls_ctx ) ;
2018-04-30 10:16:16 +03:00
spin_lock_irqsave ( & ctx - > lock , flags ) ;
info = ctx - > retransmit_hint ;
2019-09-02 21:31:06 -07:00
if ( info & & ! before ( acked_seq , info - > end_seq ) )
2018-04-30 10:16:16 +03:00
ctx - > retransmit_hint = NULL ;
list_for_each_entry_safe ( info , temp , & ctx - > records_list , list ) {
if ( before ( acked_seq , info - > end_seq ) )
break ;
list_del ( & info - > list ) ;
destroy_record ( info ) ;
deleted_records + + ;
}
ctx - > unacked_record_sn + = deleted_records ;
spin_unlock_irqrestore ( & ctx - > lock , flags ) ;
}
/* At this point, there should be no references on this
* socket and no in - flight SKBs associated with this
* socket , so it is safe to free all the resources .
*/
2019-12-17 14:12:01 -08:00
void tls_device_sk_destruct ( struct sock * sk )
2018-04-30 10:16:16 +03:00
{
struct tls_context * tls_ctx = tls_get_ctx ( sk ) ;
2018-07-13 14:33:39 +03:00
struct tls_offload_context_tx * ctx = tls_offload_ctx_tx ( tls_ctx ) ;
2018-04-30 10:16:16 +03:00
2018-07-13 14:33:43 +03:00
tls_ctx - > sk_destruct ( sk ) ;
2018-04-30 10:16:16 +03:00
2018-07-13 14:33:43 +03:00
if ( tls_ctx - > tx_conf = = TLS_HW ) {
if ( ctx - > open_record )
destroy_record ( ctx - > open_record ) ;
delete_all_records ( ctx ) ;
crypto_free_aead ( ctx - > aead_send ) ;
clean_acked_data_disable ( inet_csk ( sk ) ) ;
}
2018-04-30 10:16:16 +03:00
if ( refcount_dec_and_test ( & tls_ctx - > refcount ) )
tls_device_queue_ctx_destruction ( tls_ctx ) ;
}
2019-12-17 14:12:01 -08:00
EXPORT_SYMBOL_GPL ( tls_device_sk_destruct ) ;
2018-04-30 10:16:16 +03:00
2019-04-10 11:04:31 -07:00
void tls_device_free_resources_tx ( struct sock * sk )
{
struct tls_context * tls_ctx = tls_get_ctx ( sk ) ;
tls_free_partial_record ( sk , tls_ctx ) ;
}
2019-10-04 16:19:22 -07:00
void tls_offload_tx_resync_request ( struct sock * sk , u32 got_seq , u32 exp_seq )
{
struct tls_context * tls_ctx = tls_get_ctx ( sk ) ;
trace_tls_device_tx_resync_req ( sk , got_seq , exp_seq ) ;
WARN_ON ( test_and_set_bit ( TLS_TX_SYNC_SCHED , & tls_ctx - > flags ) ) ;
}
EXPORT_SYMBOL_GPL ( tls_offload_tx_resync_request ) ;
2019-06-10 21:40:09 -07:00
static void tls_device_resync_tx ( struct sock * sk , struct tls_context * tls_ctx ,
u32 seq )
{
struct net_device * netdev ;
struct sk_buff * skb ;
2019-07-08 19:53:13 -07:00
int err = 0 ;
2019-06-10 21:40:09 -07:00
u8 * rcd_sn ;
skb = tcp_write_queue_tail ( sk ) ;
if ( skb )
TCP_SKB_CB ( skb ) - > eor = 1 ;
rcd_sn = tls_ctx - > tx . rec_seq ;
2019-10-04 16:19:22 -07:00
trace_tls_device_tx_resync_send ( sk , seq , rcd_sn ) ;
2019-06-10 21:40:09 -07:00
down_read ( & device_offload_lock ) ;
netdev = tls_ctx - > netdev ;
if ( netdev )
2019-07-08 19:53:13 -07:00
err = netdev - > tlsdev_ops - > tls_dev_resync ( netdev , sk , seq ,
rcd_sn ,
TLS_OFFLOAD_CTX_DIR_TX ) ;
2019-06-10 21:40:09 -07:00
up_read ( & device_offload_lock ) ;
2019-07-08 19:53:13 -07:00
if ( err )
return ;
2019-06-10 21:40:09 -07:00
clear_bit_unlock ( TLS_TX_SYNC_SCHED , & tls_ctx - > flags ) ;
}
2018-04-30 10:16:16 +03:00
static void tls_append_frag ( struct tls_record_info * record ,
struct page_frag * pfrag ,
int size )
{
skb_frag_t * frag ;
frag = & record - > frags [ record - > num_frags - 1 ] ;
2019-07-22 20:08:26 -07:00
if ( skb_frag_page ( frag ) = = pfrag - > page & &
2019-07-30 07:40:33 -07:00
skb_frag_off ( frag ) + skb_frag_size ( frag ) = = pfrag - > offset ) {
2019-07-22 20:08:26 -07:00
skb_frag_size_add ( frag , size ) ;
2018-04-30 10:16:16 +03:00
} else {
+ + frag ;
2019-07-22 20:08:26 -07:00
__skb_frag_set_page ( frag , pfrag - > page ) ;
2019-07-30 07:40:33 -07:00
skb_frag_off_set ( frag , pfrag - > offset ) ;
2019-07-22 20:08:26 -07:00
skb_frag_size_set ( frag , size ) ;
2018-04-30 10:16:16 +03:00
+ + record - > num_frags ;
get_page ( pfrag - > page ) ;
}
pfrag - > offset + = size ;
record - > len + = size ;
}
static int tls_push_record ( struct sock * sk ,
struct tls_context * ctx ,
2018-07-13 14:33:39 +03:00
struct tls_offload_context_tx * offload_ctx ,
2018-04-30 10:16:16 +03:00
struct tls_record_info * record ,
2019-09-06 22:29:59 -07:00
int flags )
2018-04-30 10:16:16 +03:00
{
2019-02-14 07:11:35 +00:00
struct tls_prot_info * prot = & ctx - > prot_info ;
2018-04-30 10:16:16 +03:00
struct tcp_sock * tp = tcp_sk ( sk ) ;
skb_frag_t * frag ;
int i ;
record - > end_seq = tp - > write_seq + record - > len ;
2019-09-06 22:29:58 -07:00
list_add_tail_rcu ( & record - > list , & offload_ctx - > records_list ) ;
2018-04-30 10:16:16 +03:00
offload_ctx - > open_record = NULL ;
2019-06-10 21:40:09 -07:00
if ( test_bit ( TLS_TX_SYNC_SCHED , & ctx - > flags ) )
tls_device_resync_tx ( sk , ctx , tp - > write_seq ) ;
2019-06-03 15:17:05 -07:00
tls_advance_record_sn ( sk , prot , & ctx - > tx ) ;
2018-04-30 10:16:16 +03:00
for ( i = 0 ; i < record - > num_frags ; i + + ) {
frag = & record - > frags [ i ] ;
sg_unmark_end ( & offload_ctx - > sg_tx_data [ i ] ) ;
sg_set_page ( & offload_ctx - > sg_tx_data [ i ] , skb_frag_page ( frag ) ,
2019-07-30 07:40:33 -07:00
skb_frag_size ( frag ) , skb_frag_off ( frag ) ) ;
2019-07-22 20:08:26 -07:00
sk_mem_charge ( sk , skb_frag_size ( frag ) ) ;
2018-04-30 10:16:16 +03:00
get_page ( skb_frag_page ( frag ) ) ;
}
sg_mark_end ( & offload_ctx - > sg_tx_data [ record - > num_frags - 1 ] ) ;
/* all ready, send */
return tls_push_sg ( sk , ctx , offload_ctx - > sg_tx_data , 0 , flags ) ;
}
2019-09-06 22:29:59 -07:00
static int tls_device_record_close ( struct sock * sk ,
struct tls_context * ctx ,
struct tls_record_info * record ,
struct page_frag * pfrag ,
unsigned char record_type )
{
struct tls_prot_info * prot = & ctx - > prot_info ;
int ret ;
/* append tag
* device will fill in the tag , we just need to append a placeholder
* use socket memory to improve coalescing ( re - using a single buffer
* increases frag count )
* if we can ' t allocate memory now , steal some back from data
*/
if ( likely ( skb_page_frag_refill ( prot - > tag_size , pfrag ,
sk - > sk_allocation ) ) ) {
ret = 0 ;
tls_append_frag ( record , pfrag , prot - > tag_size ) ;
} else {
ret = prot - > tag_size ;
if ( record - > len < = prot - > overhead_size )
return - ENOMEM ;
}
/* fill prepend */
tls_fill_prepend ( ctx , skb_frag_address ( & record - > frags [ 0 ] ) ,
record - > len - prot - > overhead_size ,
record_type , prot - > version ) ;
return ret ;
}
2018-07-13 14:33:39 +03:00
static int tls_create_new_record ( struct tls_offload_context_tx * offload_ctx ,
2018-04-30 10:16:16 +03:00
struct page_frag * pfrag ,
size_t prepend_size )
{
struct tls_record_info * record ;
skb_frag_t * frag ;
record = kmalloc ( sizeof ( * record ) , GFP_KERNEL ) ;
if ( ! record )
return - ENOMEM ;
frag = & record - > frags [ 0 ] ;
__skb_frag_set_page ( frag , pfrag - > page ) ;
2019-07-30 07:40:33 -07:00
skb_frag_off_set ( frag , pfrag - > offset ) ;
2018-04-30 10:16:16 +03:00
skb_frag_size_set ( frag , prepend_size ) ;
get_page ( pfrag - > page ) ;
pfrag - > offset + = prepend_size ;
record - > num_frags = 1 ;
record - > len = prepend_size ;
offload_ctx - > open_record = record ;
return 0 ;
}
static int tls_do_allocation ( struct sock * sk ,
2018-07-13 14:33:39 +03:00
struct tls_offload_context_tx * offload_ctx ,
2018-04-30 10:16:16 +03:00
struct page_frag * pfrag ,
size_t prepend_size )
{
int ret ;
if ( ! offload_ctx - > open_record ) {
if ( unlikely ( ! skb_page_frag_refill ( prepend_size , pfrag ,
sk - > sk_allocation ) ) ) {
2020-03-17 18:04:39 +01:00
READ_ONCE ( sk - > sk_prot ) - > enter_memory_pressure ( sk ) ;
2018-04-30 10:16:16 +03:00
sk_stream_moderate_sndbuf ( sk ) ;
return - ENOMEM ;
}
ret = tls_create_new_record ( offload_ctx , pfrag , prepend_size ) ;
if ( ret )
return ret ;
if ( pfrag - > size > pfrag - > offset )
return 0 ;
}
if ( ! sk_page_frag_refill ( sk , pfrag ) )
return - ENOMEM ;
return 0 ;
}
2019-09-06 22:30:00 -07:00
static int tls_device_copy_data ( void * addr , size_t bytes , struct iov_iter * i )
{
size_t pre_copy , nocache ;
pre_copy = ~ ( ( unsigned long ) addr - 1 ) & ( SMP_CACHE_BYTES - 1 ) ;
if ( pre_copy ) {
pre_copy = min ( pre_copy , bytes ) ;
if ( copy_from_iter ( addr , pre_copy , i ) ! = pre_copy )
return - EFAULT ;
bytes - = pre_copy ;
addr + = pre_copy ;
}
nocache = round_down ( bytes , SMP_CACHE_BYTES ) ;
if ( copy_from_iter_nocache ( addr , nocache , i ) ! = nocache )
return - EFAULT ;
bytes - = nocache ;
addr + = nocache ;
if ( bytes & & copy_from_iter ( addr , bytes , i ) ! = bytes )
return - EFAULT ;
return 0 ;
}
2018-04-30 10:16:16 +03:00
static int tls_push_data ( struct sock * sk ,
struct iov_iter * msg_iter ,
size_t size , int flags ,
unsigned char record_type )
{
struct tls_context * tls_ctx = tls_get_ctx ( sk ) ;
2019-02-14 07:11:35 +00:00
struct tls_prot_info * prot = & tls_ctx - > prot_info ;
2018-07-13 14:33:39 +03:00
struct tls_offload_context_tx * ctx = tls_offload_ctx_tx ( tls_ctx ) ;
2018-04-30 10:16:16 +03:00
int more = flags & ( MSG_SENDPAGE_NOTLAST | MSG_MORE ) ;
struct tls_record_info * record = ctx - > open_record ;
2019-08-07 17:03:59 -07:00
int tls_push_record_flags ;
2018-04-30 10:16:16 +03:00
struct page_frag * pfrag ;
size_t orig_size = size ;
u32 max_open_record_len ;
int copy , rc = 0 ;
bool done = false ;
long timeo ;
if ( flags &
~ ( MSG_MORE | MSG_DONTWAIT | MSG_NOSIGNAL | MSG_SENDPAGE_NOTLAST ) )
2019-12-05 07:41:18 +01:00
return - EOPNOTSUPP ;
2018-04-30 10:16:16 +03:00
2019-10-06 21:09:28 -07:00
if ( unlikely ( sk - > sk_err ) )
2018-04-30 10:16:16 +03:00
return - sk - > sk_err ;
2019-08-07 17:03:59 -07:00
flags | = MSG_SENDPAGE_DECRYPTED ;
tls_push_record_flags = flags | MSG_SENDPAGE_NOTLAST ;
2018-04-30 10:16:16 +03:00
timeo = sock_sndtimeo ( sk , flags & MSG_DONTWAIT ) ;
2019-02-27 17:38:03 +02:00
if ( tls_is_partially_sent_record ( tls_ctx ) ) {
rc = tls_push_partial_record ( sk , tls_ctx , flags ) ;
if ( rc < 0 )
return rc ;
}
2018-04-30 10:16:16 +03:00
pfrag = sk_page_frag ( sk ) ;
/* TLS_HEADER_SIZE is not counted as part of the TLS record, and
* we need to leave room for an authentication tag .
*/
max_open_record_len = TLS_MAX_PAYLOAD_SIZE +
2019-02-14 07:11:35 +00:00
prot - > prepend_size ;
2018-04-30 10:16:16 +03:00
do {
2019-10-06 21:09:29 -07:00
rc = tls_do_allocation ( sk , ctx , pfrag , prot - > prepend_size ) ;
if ( unlikely ( rc ) ) {
2018-04-30 10:16:16 +03:00
rc = sk_stream_wait_memory ( sk , & timeo ) ;
if ( ! rc )
continue ;
record = ctx - > open_record ;
if ( ! record )
break ;
handle_error :
if ( record_type ! = TLS_RECORD_TYPE_DATA ) {
/* avoid sending partial
* record with type ! =
* application_data
*/
size = orig_size ;
destroy_record ( record ) ;
ctx - > open_record = NULL ;
2019-02-14 07:11:35 +00:00
} else if ( record - > len > prot - > prepend_size ) {
2018-04-30 10:16:16 +03:00
goto last_record ;
}
break ;
}
record = ctx - > open_record ;
copy = min_t ( size_t , size , ( pfrag - > size - pfrag - > offset ) ) ;
copy = min_t ( size_t , copy , ( max_open_record_len - record - > len ) ) ;
2019-09-06 22:30:00 -07:00
rc = tls_device_copy_data ( page_address ( pfrag - > page ) +
pfrag - > offset , copy , msg_iter ) ;
if ( rc )
2018-04-30 10:16:16 +03:00
goto handle_error ;
tls_append_frag ( record , pfrag , copy ) ;
size - = copy ;
if ( ! size ) {
last_record :
tls_push_record_flags = flags ;
if ( more ) {
tls_ctx - > pending_open_record_frags =
2018-10-13 02:45:59 +02:00
! ! record - > num_frags ;
2018-04-30 10:16:16 +03:00
break ;
}
done = true ;
}
if ( done | | record - > len > = max_open_record_len | |
( record - > num_frags > = MAX_SKB_FRAGS - 1 ) ) {
2019-09-06 22:29:59 -07:00
rc = tls_device_record_close ( sk , tls_ctx , record ,
pfrag , record_type ) ;
if ( rc ) {
if ( rc > 0 ) {
size + = rc ;
} else {
size = orig_size ;
destroy_record ( record ) ;
ctx - > open_record = NULL ;
break ;
}
}
2018-04-30 10:16:16 +03:00
rc = tls_push_record ( sk ,
tls_ctx ,
ctx ,
record ,
2019-09-06 22:29:59 -07:00
tls_push_record_flags ) ;
2018-04-30 10:16:16 +03:00
if ( rc < 0 )
break ;
}
} while ( ! done ) ;
if ( orig_size - size > 0 )
rc = orig_size - size ;
return rc ;
}
int tls_device_sendmsg ( struct sock * sk , struct msghdr * msg , size_t size )
{
unsigned char record_type = TLS_RECORD_TYPE_DATA ;
2019-11-05 14:24:35 -08:00
struct tls_context * tls_ctx = tls_get_ctx ( sk ) ;
2018-04-30 10:16:16 +03:00
int rc ;
2019-11-05 14:24:35 -08:00
mutex_lock ( & tls_ctx - > tx_lock ) ;
2018-04-30 10:16:16 +03:00
lock_sock ( sk ) ;
if ( unlikely ( msg - > msg_controllen ) ) {
rc = tls_proccess_cmsg ( sk , msg , & record_type ) ;
if ( rc )
goto out ;
}
rc = tls_push_data ( sk , & msg - > msg_iter , size ,
msg - > msg_flags , record_type ) ;
out :
release_sock ( sk ) ;
2019-11-05 14:24:35 -08:00
mutex_unlock ( & tls_ctx - > tx_lock ) ;
2018-04-30 10:16:16 +03:00
return rc ;
}
int tls_device_sendpage ( struct sock * sk , struct page * page ,
int offset , size_t size , int flags )
{
2019-11-05 14:24:35 -08:00
struct tls_context * tls_ctx = tls_get_ctx ( sk ) ;
2018-04-30 10:16:16 +03:00
struct iov_iter msg_iter ;
char * kaddr = kmap ( page ) ;
struct kvec iov ;
int rc ;
if ( flags & MSG_SENDPAGE_NOTLAST )
flags | = MSG_MORE ;
2019-11-05 14:24:35 -08:00
mutex_lock ( & tls_ctx - > tx_lock ) ;
2018-04-30 10:16:16 +03:00
lock_sock ( sk ) ;
if ( flags & MSG_OOB ) {
2019-12-05 07:41:18 +01:00
rc = - EOPNOTSUPP ;
2018-04-30 10:16:16 +03:00
goto out ;
}
iov . iov_base = kaddr + offset ;
iov . iov_len = size ;
2018-10-20 00:57:56 +01:00
iov_iter_kvec ( & msg_iter , WRITE , & iov , 1 , size ) ;
2018-04-30 10:16:16 +03:00
rc = tls_push_data ( sk , & msg_iter , size ,
flags , TLS_RECORD_TYPE_DATA ) ;
kunmap ( page ) ;
out :
release_sock ( sk ) ;
2019-11-05 14:24:35 -08:00
mutex_unlock ( & tls_ctx - > tx_lock ) ;
2018-04-30 10:16:16 +03:00
return rc ;
}
2018-07-13 14:33:39 +03:00
struct tls_record_info * tls_get_record ( struct tls_offload_context_tx * context ,
2018-04-30 10:16:16 +03:00
u32 seq , u64 * p_record_sn )
{
u64 record_sn = context - > hint_record_sn ;
2020-02-19 09:40:22 +05:30
struct tls_record_info * info , * last ;
2018-04-30 10:16:16 +03:00
info = context - > retransmit_hint ;
if ( ! info | |
before ( seq , info - > end_seq - info - > len ) ) {
/* if retransmit_hint is irrelevant start
* from the beggining of the list
*/
2019-09-06 22:29:58 -07:00
info = list_first_entry_or_null ( & context - > records_list ,
struct tls_record_info , list ) ;
if ( ! info )
return NULL ;
2020-02-19 09:40:22 +05:30
/* send the start_marker record if seq number is before the
* tls offload start marker sequence number . This record is
* required to handle TCP packets which are before TLS offload
* started .
* And if it ' s not start marker , look if this seq number
* belongs to the list .
*/
if ( likely ( ! tls_record_is_start_marker ( info ) ) ) {
/* we have the first record, get the last record to see
* if this seq number belongs to the list .
*/
last = list_last_entry ( & context - > records_list ,
struct tls_record_info , list ) ;
if ( ! between ( seq , tls_record_start_seq ( info ) ,
last - > end_seq ) )
return NULL ;
}
2018-04-30 10:16:16 +03:00
record_sn = context - > unacked_record_sn ;
}
2019-09-06 22:29:58 -07:00
/* We just need the _rcu for the READ_ONCE() */
rcu_read_lock ( ) ;
list_for_each_entry_from_rcu ( info , & context - > records_list , list ) {
2018-04-30 10:16:16 +03:00
if ( before ( seq , info - > end_seq ) ) {
if ( ! context - > retransmit_hint | |
after ( info - > end_seq ,
context - > retransmit_hint - > end_seq ) ) {
context - > hint_record_sn = record_sn ;
context - > retransmit_hint = info ;
}
* p_record_sn = record_sn ;
2019-09-06 22:29:58 -07:00
goto exit_rcu_unlock ;
2018-04-30 10:16:16 +03:00
}
record_sn + + ;
}
2019-09-06 22:29:58 -07:00
info = NULL ;
2018-04-30 10:16:16 +03:00
2019-09-06 22:29:58 -07:00
exit_rcu_unlock :
rcu_read_unlock ( ) ;
return info ;
2018-04-30 10:16:16 +03:00
}
EXPORT_SYMBOL ( tls_get_record ) ;
static int tls_device_push_pending_record ( struct sock * sk , int flags )
{
struct iov_iter msg_iter ;
2018-10-20 00:57:56 +01:00
iov_iter_kvec ( & msg_iter , WRITE , NULL , 0 , 0 ) ;
2018-04-30 10:16:16 +03:00
return tls_push_data ( sk , & msg_iter , 0 , flags , TLS_RECORD_TYPE_DATA ) ;
}
2019-02-27 17:38:04 +02:00
void tls_device_write_space ( struct sock * sk , struct tls_context * ctx )
{
2019-11-05 14:24:34 -08:00
if ( tls_is_partially_sent_record ( ctx ) ) {
2019-02-27 17:38:04 +02:00
gfp_t sk_allocation = sk - > sk_allocation ;
2019-11-05 14:24:34 -08:00
WARN_ON_ONCE ( sk - > sk_write_pending ) ;
2019-02-27 17:38:04 +02:00
sk - > sk_allocation = GFP_ATOMIC ;
2019-08-07 17:03:59 -07:00
tls_push_partial_record ( sk , ctx ,
MSG_DONTWAIT | MSG_NOSIGNAL |
MSG_SENDPAGE_DECRYPTED ) ;
2019-02-27 17:38:04 +02:00
sk - > sk_allocation = sk_allocation ;
}
}
2019-06-04 12:00:12 -07:00
static void tls_device_resync_rx ( struct tls_context * tls_ctx ,
2019-06-10 21:40:00 -07:00
struct sock * sk , u32 seq , u8 * rcd_sn )
2019-06-04 12:00:12 -07:00
{
2019-10-04 16:19:22 -07:00
struct tls_offload_context_rx * rx_ctx = tls_offload_ctx_rx ( tls_ctx ) ;
2019-06-04 12:00:12 -07:00
struct net_device * netdev ;
if ( WARN_ON ( test_and_set_bit ( TLS_RX_SYNC_RUNNING , & tls_ctx - > flags ) ) )
return ;
2019-10-04 16:19:22 -07:00
trace_tls_device_rx_resync_send ( sk , seq , rcd_sn , rx_ctx - > resync_type ) ;
2019-06-04 12:00:12 -07:00
netdev = READ_ONCE ( tls_ctx - > netdev ) ;
if ( netdev )
2019-06-10 21:40:08 -07:00
netdev - > tlsdev_ops - > tls_dev_resync ( netdev , sk , seq , rcd_sn ,
TLS_OFFLOAD_CTX_DIR_RX ) ;
2019-06-04 12:00:12 -07:00
clear_bit_unlock ( TLS_RX_SYNC_RUNNING , & tls_ctx - > flags ) ;
2019-10-04 16:19:27 -07:00
TLS_INC_STATS ( sock_net ( sk ) , LINUX_MIB_TLSRXDEVICERESYNC ) ;
2019-06-04 12:00:12 -07:00
}
2019-06-10 21:40:02 -07:00
void tls_device_rx_resync_new_rec ( struct sock * sk , u32 rcd_len , u32 seq )
2018-07-13 14:33:43 +03:00
{
struct tls_context * tls_ctx = tls_get_ctx ( sk ) ;
struct tls_offload_context_rx * rx_ctx ;
2019-06-10 21:40:02 -07:00
u8 rcd_sn [ TLS_MAX_REC_SEQ_SIZE ] ;
2019-10-04 16:19:22 -07:00
u32 sock_data , is_req_pending ;
2019-06-10 21:40:02 -07:00
struct tls_prot_info * prot ;
2018-07-13 14:33:43 +03:00
s64 resync_req ;
u32 req_seq ;
if ( tls_ctx - > rx_conf ! = TLS_HW )
return ;
2019-06-10 21:40:02 -07:00
prot = & tls_ctx - > prot_info ;
2018-07-13 14:33:43 +03:00
rx_ctx = tls_offload_ctx_rx ( tls_ctx ) ;
2019-06-10 21:40:02 -07:00
memcpy ( rcd_sn , tls_ctx - > rx . rec_seq , prot - > rec_seq_size ) ;
switch ( rx_ctx - > resync_type ) {
case TLS_OFFLOAD_SYNC_TYPE_DRIVER_REQ :
resync_req = atomic64_read ( & rx_ctx - > resync_req ) ;
req_seq = resync_req > > 32 ;
seq + = TLS_HEADER_SIZE - 1 ;
is_req_pending = resync_req ;
if ( likely ( ! is_req_pending ) | | req_seq ! = seq | |
! atomic64_try_cmpxchg ( & rx_ctx - > resync_req , & resync_req , 0 ) )
return ;
break ;
case TLS_OFFLOAD_SYNC_TYPE_CORE_NEXT_HINT :
if ( likely ( ! rx_ctx - > resync_nh_do_now ) )
return ;
/* head of next rec is already in, note that the sock_inq will
* include the currently parsed message when called from parser
*/
2019-10-04 16:19:22 -07:00
sock_data = tcp_inq ( sk ) ;
if ( sock_data > rcd_len ) {
trace_tls_device_rx_resync_nh_delay ( sk , sock_data ,
rcd_len ) ;
2019-06-10 21:40:02 -07:00
return ;
2019-10-04 16:19:22 -07:00
}
2019-06-10 21:40:02 -07:00
rx_ctx - > resync_nh_do_now = 0 ;
seq + = rcd_len ;
tls_bigint_increment ( rcd_sn , prot - > rec_seq_size ) ;
break ;
}
tls_device_resync_rx ( tls_ctx , sk , seq , rcd_sn ) ;
}
static void tls_device_core_ctrl_rx_resync ( struct tls_context * tls_ctx ,
struct tls_offload_context_rx * ctx ,
struct sock * sk , struct sk_buff * skb )
{
struct strp_msg * rxm ;
/* device will request resyncs by itself based on stream scan */
if ( ctx - > resync_type ! = TLS_OFFLOAD_SYNC_TYPE_CORE_NEXT_HINT )
return ;
/* already scheduled */
if ( ctx - > resync_nh_do_now )
return ;
/* seen decrypted fragments since last fully-failed record */
if ( ctx - > resync_nh_reset ) {
ctx - > resync_nh_reset = 0 ;
ctx - > resync_nh . decrypted_failed = 1 ;
ctx - > resync_nh . decrypted_tgt = TLS_DEVICE_RESYNC_NH_START_IVAL ;
return ;
}
if ( + + ctx - > resync_nh . decrypted_failed < = ctx - > resync_nh . decrypted_tgt )
return ;
/* doing resync, bump the next target in case it fails */
if ( ctx - > resync_nh . decrypted_tgt < TLS_DEVICE_RESYNC_NH_MAX_IVAL )
ctx - > resync_nh . decrypted_tgt * = 2 ;
else
ctx - > resync_nh . decrypted_tgt + = TLS_DEVICE_RESYNC_NH_MAX_IVAL ;
rxm = strp_msg ( skb ) ;
/* head of next rec is already in, parser will sync for us */
if ( tcp_inq ( sk ) > rxm - > full_len ) {
2019-10-04 16:19:22 -07:00
trace_tls_device_rx_resync_nh_schedule ( sk ) ;
2019-06-10 21:40:02 -07:00
ctx - > resync_nh_do_now = 1 ;
} else {
struct tls_prot_info * prot = & tls_ctx - > prot_info ;
u8 rcd_sn [ TLS_MAX_REC_SEQ_SIZE ] ;
memcpy ( rcd_sn , tls_ctx - > rx . rec_seq , prot - > rec_seq_size ) ;
tls_bigint_increment ( rcd_sn , prot - > rec_seq_size ) ;
tls_device_resync_rx ( tls_ctx , sk , tcp_sk ( sk ) - > copied_seq ,
rcd_sn ) ;
}
2018-07-13 14:33:43 +03:00
}
static int tls_device_reencrypt ( struct sock * sk , struct sk_buff * skb )
{
struct strp_msg * rxm = strp_msg ( skb ) ;
2019-04-25 17:35:10 -07:00
int err = 0 , offset = rxm - > offset , copy , nsg , data_len , pos ;
2018-07-13 14:33:43 +03:00
struct sk_buff * skb_iter , * unused ;
struct scatterlist sg [ 1 ] ;
char * orig_buf , * buf ;
orig_buf = kmalloc ( rxm - > full_len + TLS_HEADER_SIZE +
TLS_CIPHER_AES_GCM_128_IV_SIZE , sk - > sk_allocation ) ;
if ( ! orig_buf )
return - ENOMEM ;
buf = orig_buf ;
nsg = skb_cow_data ( skb , 0 , & unused ) ;
if ( unlikely ( nsg < 0 ) ) {
err = nsg ;
goto free_buf ;
}
sg_init_table ( sg , 1 ) ;
sg_set_buf ( & sg [ 0 ] , buf ,
rxm - > full_len + TLS_HEADER_SIZE +
TLS_CIPHER_AES_GCM_128_IV_SIZE ) ;
2019-06-03 15:16:59 -07:00
err = skb_copy_bits ( skb , offset , buf ,
TLS_HEADER_SIZE + TLS_CIPHER_AES_GCM_128_IV_SIZE ) ;
if ( err )
goto free_buf ;
2018-07-13 14:33:43 +03:00
/* We are interested only in the decrypted data not the auth */
err = decrypt_skb ( sk , skb , sg ) ;
if ( err ! = - EBADMSG )
goto free_buf ;
else
err = 0 ;
2019-04-25 17:35:10 -07:00
data_len = rxm - > full_len - TLS_CIPHER_AES_GCM_128_TAG_SIZE ;
2018-07-13 14:33:43 +03:00
2019-04-25 17:35:09 -07:00
if ( skb_pagelen ( skb ) > offset ) {
2019-04-25 17:35:10 -07:00
copy = min_t ( int , skb_pagelen ( skb ) - offset , data_len ) ;
2018-07-13 14:33:43 +03:00
2019-06-03 15:16:59 -07:00
if ( skb - > decrypted ) {
err = skb_store_bits ( skb , offset , buf , copy ) ;
if ( err )
goto free_buf ;
}
2018-07-13 14:33:43 +03:00
2019-04-25 17:35:09 -07:00
offset + = copy ;
buf + = copy ;
}
2018-07-13 14:33:43 +03:00
2019-04-25 17:35:10 -07:00
pos = skb_pagelen ( skb ) ;
2018-07-13 14:33:43 +03:00
skb_walk_frags ( skb , skb_iter ) {
2019-04-25 17:35:10 -07:00
int frag_pos ;
/* Practically all frags must belong to msg if reencrypt
* is needed with current strparser and coalescing logic ,
* but strparser may " get optimized " , so let ' s be safe .
*/
if ( pos + skb_iter - > len < = offset )
goto done_with_frag ;
if ( pos > = data_len + rxm - > offset )
break ;
frag_pos = offset - pos ;
copy = min_t ( int , skb_iter - > len - frag_pos ,
data_len + rxm - > offset - offset ) ;
2018-07-13 14:33:43 +03:00
2019-06-03 15:16:59 -07:00
if ( skb_iter - > decrypted ) {
err = skb_store_bits ( skb_iter , frag_pos , buf , copy ) ;
if ( err )
goto free_buf ;
}
2018-07-13 14:33:43 +03:00
offset + = copy ;
buf + = copy ;
2019-04-25 17:35:10 -07:00
done_with_frag :
pos + = skb_iter - > len ;
2018-07-13 14:33:43 +03:00
}
free_buf :
kfree ( orig_buf ) ;
return err ;
}
2019-10-06 21:09:30 -07:00
int tls_device_decrypted ( struct sock * sk , struct tls_context * tls_ctx ,
struct sk_buff * skb , struct strp_msg * rxm )
2018-07-13 14:33:43 +03:00
{
struct tls_offload_context_rx * ctx = tls_offload_ctx_rx ( tls_ctx ) ;
int is_decrypted = skb - > decrypted ;
int is_encrypted = ! is_decrypted ;
struct sk_buff * skb_iter ;
/* Check if all the data is decrypted already */
skb_walk_frags ( skb , skb_iter ) {
is_decrypted & = skb_iter - > decrypted ;
is_encrypted & = ! skb_iter - > decrypted ;
}
2019-10-04 16:19:23 -07:00
trace_tls_device_decrypted ( sk , tcp_sk ( sk ) - > copied_seq - rxm - > full_len ,
tls_ctx - > rx . rec_seq , rxm - > full_len ,
is_encrypted , is_decrypted ) ;
2018-07-13 14:33:43 +03:00
ctx - > sw . decrypted | = is_decrypted ;
2019-06-10 21:40:02 -07:00
/* Return immediately if the record is either entirely plaintext or
2018-07-13 14:33:43 +03:00
* entirely ciphertext . Otherwise handle reencrypt partially decrypted
* record .
*/
2019-06-10 21:40:02 -07:00
if ( is_decrypted ) {
ctx - > resync_nh_reset = 1 ;
return 0 ;
}
if ( is_encrypted ) {
tls_device_core_ctrl_rx_resync ( tls_ctx , ctx , sk , skb ) ;
return 0 ;
}
ctx - > resync_nh_reset = 1 ;
return tls_device_reencrypt ( sk , skb ) ;
2018-07-13 14:33:43 +03:00
}
2019-04-25 12:32:02 -07:00
static void tls_device_attach ( struct tls_context * ctx , struct sock * sk ,
struct net_device * netdev )
{
if ( sk - > sk_destruct ! = tls_device_sk_destruct ) {
refcount_set ( & ctx - > refcount , 1 ) ;
dev_hold ( netdev ) ;
ctx - > netdev = netdev ;
spin_lock_irq ( & tls_device_lock ) ;
list_add_tail ( & ctx - > list , & tls_device_list ) ;
spin_unlock_irq ( & tls_device_lock ) ;
ctx - > sk_destruct = sk - > sk_destruct ;
2019-12-17 14:12:01 -08:00
smp_store_release ( & sk - > sk_destruct , tls_device_sk_destruct ) ;
2019-04-25 12:32:02 -07:00
}
}
2018-04-30 10:16:16 +03:00
int tls_set_device_offload ( struct sock * sk , struct tls_context * ctx )
{
u16 nonce_size , tag_size , iv_size , rec_seq_size ;
2019-02-14 07:11:35 +00:00
struct tls_context * tls_ctx = tls_get_ctx ( sk ) ;
struct tls_prot_info * prot = & tls_ctx - > prot_info ;
2018-04-30 10:16:16 +03:00
struct tls_record_info * start_marker_record ;
2018-07-13 14:33:39 +03:00
struct tls_offload_context_tx * offload_ctx ;
2018-04-30 10:16:16 +03:00
struct tls_crypto_info * crypto_info ;
struct net_device * netdev ;
char * iv , * rec_seq ;
struct sk_buff * skb ;
__be64 rcd_sn ;
2019-09-02 21:31:03 -07:00
int rc ;
2018-04-30 10:16:16 +03:00
if ( ! ctx )
2019-09-02 21:31:03 -07:00
return - EINVAL ;
2018-04-30 10:16:16 +03:00
2019-09-02 21:31:03 -07:00
if ( ctx - > priv_ctx_tx )
return - EEXIST ;
2018-04-30 10:16:16 +03:00
start_marker_record = kmalloc ( sizeof ( * start_marker_record ) , GFP_KERNEL ) ;
2019-09-02 21:31:03 -07:00
if ( ! start_marker_record )
return - ENOMEM ;
2018-04-30 10:16:16 +03:00
2018-07-13 14:33:39 +03:00
offload_ctx = kzalloc ( TLS_OFFLOAD_CONTEXT_SIZE_TX , GFP_KERNEL ) ;
2018-04-30 10:16:16 +03:00
if ( ! offload_ctx ) {
rc = - ENOMEM ;
goto free_marker_record ;
}
2018-09-12 17:44:42 +02:00
crypto_info = & ctx - > crypto_send . info ;
2019-06-28 16:07:59 -07:00
if ( crypto_info - > version ! = TLS_1_2_VERSION ) {
rc = - EOPNOTSUPP ;
goto free_offload_ctx ;
}
2018-04-30 10:16:16 +03:00
switch ( crypto_info - > cipher_type ) {
case TLS_CIPHER_AES_GCM_128 :
nonce_size = TLS_CIPHER_AES_GCM_128_IV_SIZE ;
tag_size = TLS_CIPHER_AES_GCM_128_TAG_SIZE ;
iv_size = TLS_CIPHER_AES_GCM_128_IV_SIZE ;
iv = ( ( struct tls12_crypto_info_aes_gcm_128 * ) crypto_info ) - > iv ;
rec_seq_size = TLS_CIPHER_AES_GCM_128_REC_SEQ_SIZE ;
rec_seq =
( ( struct tls12_crypto_info_aes_gcm_128 * ) crypto_info ) - > rec_seq ;
break ;
default :
rc = - EINVAL ;
goto free_offload_ctx ;
}
2019-06-10 21:40:00 -07:00
/* Sanity-check the rec_seq_size for stack allocations */
if ( rec_seq_size > TLS_MAX_REC_SEQ_SIZE ) {
rc = - EINVAL ;
goto free_offload_ctx ;
}
2019-07-08 19:53:15 -07:00
prot - > version = crypto_info - > version ;
prot - > cipher_type = crypto_info - > cipher_type ;
2019-02-14 07:11:35 +00:00
prot - > prepend_size = TLS_HEADER_SIZE + nonce_size ;
prot - > tag_size = tag_size ;
prot - > overhead_size = prot - > prepend_size + prot - > tag_size ;
prot - > iv_size = iv_size ;
2018-04-30 10:16:16 +03:00
ctx - > tx . iv = kmalloc ( iv_size + TLS_CIPHER_AES_GCM_128_SALT_SIZE ,
GFP_KERNEL ) ;
if ( ! ctx - > tx . iv ) {
rc = - ENOMEM ;
goto free_offload_ctx ;
}
memcpy ( ctx - > tx . iv + TLS_CIPHER_AES_GCM_128_SALT_SIZE , iv , iv_size ) ;
2019-02-14 07:11:35 +00:00
prot - > rec_seq_size = rec_seq_size ;
2018-08-01 00:50:24 +08:00
ctx - > tx . rec_seq = kmemdup ( rec_seq , rec_seq_size , GFP_KERNEL ) ;
2018-04-30 10:16:16 +03:00
if ( ! ctx - > tx . rec_seq ) {
rc = - ENOMEM ;
goto free_iv ;
}
rc = tls_sw_fallback_init ( sk , offload_ctx , crypto_info ) ;
if ( rc )
goto free_rec_seq ;
/* start at rec_seq - 1 to account for the start marker record */
memcpy ( & rcd_sn , ctx - > tx . rec_seq , sizeof ( rcd_sn ) ) ;
offload_ctx - > unacked_record_sn = be64_to_cpu ( rcd_sn ) - 1 ;
start_marker_record - > end_seq = tcp_sk ( sk ) - > write_seq ;
start_marker_record - > len = 0 ;
start_marker_record - > num_frags = 0 ;
INIT_LIST_HEAD ( & offload_ctx - > records_list ) ;
list_add_tail ( & start_marker_record - > list , & offload_ctx - > records_list ) ;
spin_lock_init ( & offload_ctx - > lock ) ;
2018-05-10 16:27:25 +03:00
sg_init_table ( offload_ctx - > sg_tx_data ,
ARRAY_SIZE ( offload_ctx - > sg_tx_data ) ) ;
2018-04-30 10:16:16 +03:00
clean_acked_data_enable ( inet_csk ( sk ) , & tls_icsk_clean_acked ) ;
ctx - > push_pending_record = tls_device_push_pending_record ;
/* TLS offload is greatly simplified if we don't send
* SKBs where only part of the payload needs to be encrypted .
* So mark the last skb in the write queue as end of record .
*/
skb = tcp_write_queue_tail ( sk ) ;
if ( skb )
TCP_SKB_CB ( skb ) - > eor = 1 ;
netdev = get_netdev_for_sock ( sk ) ;
if ( ! netdev ) {
pr_err_ratelimited ( " %s: netdev not found \n " , __func__ ) ;
rc = - EINVAL ;
2019-09-02 21:31:04 -07:00
goto disable_cad ;
2018-04-30 10:16:16 +03:00
}
if ( ! ( netdev - > features & NETIF_F_HW_TLS_TX ) ) {
2019-12-05 07:41:18 +01:00
rc = - EOPNOTSUPP ;
2018-04-30 10:16:16 +03:00
goto release_netdev ;
}
/* Avoid offloading if the device is down
* We don ' t want to offload new flows after
* the NETDEV_DOWN event
2019-09-02 21:31:04 -07:00
*
* device_offload_lock is taken in tls_devices ' s NETDEV_DOWN
* handler thus protecting from the device going down before
* ctx was added to tls_device_list .
2018-04-30 10:16:16 +03:00
*/
2019-09-02 21:31:04 -07:00
down_read ( & device_offload_lock ) ;
2018-04-30 10:16:16 +03:00
if ( ! ( netdev - > flags & IFF_UP ) ) {
rc = - EINVAL ;
2019-09-02 21:31:04 -07:00
goto release_lock ;
2018-04-30 10:16:16 +03:00
}
ctx - > priv_ctx_tx = offload_ctx ;
rc = netdev - > tlsdev_ops - > tls_dev_add ( netdev , sk , TLS_OFFLOAD_CTX_DIR_TX ,
2018-09-12 17:44:42 +02:00
& ctx - > crypto_send . info ,
2018-04-30 10:16:16 +03:00
tcp_sk ( sk ) - > write_seq ) ;
2019-10-04 16:19:22 -07:00
trace_tls_device_offload_set ( sk , TLS_OFFLOAD_CTX_DIR_TX ,
tcp_sk ( sk ) - > write_seq , rec_seq , rc ) ;
2018-04-30 10:16:16 +03:00
if ( rc )
2019-09-02 21:31:04 -07:00
goto release_lock ;
2018-04-30 10:16:16 +03:00
2018-07-13 14:33:43 +03:00
tls_device_attach ( ctx , sk , netdev ) ;
2019-09-02 21:31:04 -07:00
up_read ( & device_offload_lock ) ;
2018-04-30 10:16:16 +03:00
/* following this assignment tls_is_sk_tx_device_offloaded
* will return true and the context might be accessed
* by the netdev ' s xmit function .
*/
2018-07-13 14:33:43 +03:00
smp_store_release ( & sk - > sk_validate_xmit_skb , tls_validate_xmit_skb ) ;
dev_put ( netdev ) ;
2019-09-02 21:31:03 -07:00
return 0 ;
2018-04-30 10:16:16 +03:00
release_lock :
up_read ( & device_offload_lock ) ;
2019-09-02 21:31:04 -07:00
release_netdev :
dev_put ( netdev ) ;
disable_cad :
2018-04-30 10:16:16 +03:00
clean_acked_data_disable ( inet_csk ( sk ) ) ;
crypto_free_aead ( offload_ctx - > aead_send ) ;
free_rec_seq :
kfree ( ctx - > tx . rec_seq ) ;
free_iv :
kfree ( ctx - > tx . iv ) ;
free_offload_ctx :
kfree ( offload_ctx ) ;
ctx - > priv_ctx_tx = NULL ;
free_marker_record :
kfree ( start_marker_record ) ;
return rc ;
}
2018-07-13 14:33:43 +03:00
int tls_set_device_offload_rx ( struct sock * sk , struct tls_context * ctx )
{
2019-10-04 16:19:22 -07:00
struct tls12_crypto_info_aes_gcm_128 * info ;
2018-07-13 14:33:43 +03:00
struct tls_offload_context_rx * context ;
struct net_device * netdev ;
int rc = 0 ;
2019-06-28 16:07:59 -07:00
if ( ctx - > crypto_recv . info . version ! = TLS_1_2_VERSION )
return - EOPNOTSUPP ;
2018-07-13 14:33:43 +03:00
netdev = get_netdev_for_sock ( sk ) ;
if ( ! netdev ) {
pr_err_ratelimited ( " %s: netdev not found \n " , __func__ ) ;
2019-09-02 21:31:04 -07:00
return - EINVAL ;
2018-07-13 14:33:43 +03:00
}
if ( ! ( netdev - > features & NETIF_F_HW_TLS_RX ) ) {
2019-12-05 07:41:18 +01:00
rc = - EOPNOTSUPP ;
2018-07-13 14:33:43 +03:00
goto release_netdev ;
}
/* Avoid offloading if the device is down
* We don ' t want to offload new flows after
* the NETDEV_DOWN event
2019-09-02 21:31:04 -07:00
*
* device_offload_lock is taken in tls_devices ' s NETDEV_DOWN
* handler thus protecting from the device going down before
* ctx was added to tls_device_list .
2018-07-13 14:33:43 +03:00
*/
2019-09-02 21:31:04 -07:00
down_read ( & device_offload_lock ) ;
2018-07-13 14:33:43 +03:00
if ( ! ( netdev - > flags & IFF_UP ) ) {
rc = - EINVAL ;
2019-09-02 21:31:04 -07:00
goto release_lock ;
2018-07-13 14:33:43 +03:00
}
context = kzalloc ( TLS_OFFLOAD_CONTEXT_SIZE_RX , GFP_KERNEL ) ;
if ( ! context ) {
rc = - ENOMEM ;
2019-09-02 21:31:04 -07:00
goto release_lock ;
2018-07-13 14:33:43 +03:00
}
2019-06-10 21:40:02 -07:00
context - > resync_nh_reset = 1 ;
2018-07-13 14:33:43 +03:00
ctx - > priv_ctx_rx = context ;
rc = tls_set_sw_offload ( sk , ctx , 0 ) ;
if ( rc )
goto release_ctx ;
rc = netdev - > tlsdev_ops - > tls_dev_add ( netdev , sk , TLS_OFFLOAD_CTX_DIR_RX ,
2018-09-12 17:44:42 +02:00
& ctx - > crypto_recv . info ,
2018-07-13 14:33:43 +03:00
tcp_sk ( sk ) - > copied_seq ) ;
2019-10-04 16:19:22 -07:00
info = ( void * ) & ctx - > crypto_recv . info ;
trace_tls_device_offload_set ( sk , TLS_OFFLOAD_CTX_DIR_RX ,
tcp_sk ( sk ) - > copied_seq , info - > rec_seq , rc ) ;
2019-04-25 12:32:01 -07:00
if ( rc )
2018-07-13 14:33:43 +03:00
goto free_sw_resources ;
tls_device_attach ( ctx , sk , netdev ) ;
2019-09-02 21:31:03 -07:00
up_read ( & device_offload_lock ) ;
dev_put ( netdev ) ;
return 0 ;
2018-07-13 14:33:43 +03:00
free_sw_resources :
2019-04-19 16:51:38 -07:00
up_read ( & device_offload_lock ) ;
2018-07-13 14:33:43 +03:00
tls_sw_free_resources_rx ( sk ) ;
2019-04-19 16:51:38 -07:00
down_read ( & device_offload_lock ) ;
2018-07-13 14:33:43 +03:00
release_ctx :
ctx - > priv_ctx_rx = NULL ;
release_lock :
up_read ( & device_offload_lock ) ;
2019-09-02 21:31:04 -07:00
release_netdev :
dev_put ( netdev ) ;
2018-07-13 14:33:43 +03:00
return rc ;
}
void tls_device_offload_cleanup_rx ( struct sock * sk )
{
struct tls_context * tls_ctx = tls_get_ctx ( sk ) ;
struct net_device * netdev ;
down_read ( & device_offload_lock ) ;
netdev = tls_ctx - > netdev ;
if ( ! netdev )
goto out ;
netdev - > tlsdev_ops - > tls_dev_del ( netdev , tls_ctx ,
TLS_OFFLOAD_CTX_DIR_RX ) ;
if ( tls_ctx - > tx_conf ! = TLS_HW ) {
dev_put ( netdev ) ;
tls_ctx - > netdev = NULL ;
}
out :
up_read ( & device_offload_lock ) ;
tls_sw_release_resources_rx ( sk ) ;
}
2018-04-30 10:16:16 +03:00
static int tls_device_down ( struct net_device * netdev )
{
struct tls_context * ctx , * tmp ;
unsigned long flags ;
LIST_HEAD ( list ) ;
/* Request a write lock to block new offload attempts */
down_write ( & device_offload_lock ) ;
spin_lock_irqsave ( & tls_device_lock , flags ) ;
list_for_each_entry_safe ( ctx , tmp , & tls_device_list , list ) {
if ( ctx - > netdev ! = netdev | |
! refcount_inc_not_zero ( & ctx - > refcount ) )
continue ;
list_move ( & ctx - > list , & list ) ;
}
spin_unlock_irqrestore ( & tls_device_lock , flags ) ;
list_for_each_entry_safe ( ctx , tmp , & list , list ) {
2018-07-13 14:33:43 +03:00
if ( ctx - > tx_conf = = TLS_HW )
netdev - > tlsdev_ops - > tls_dev_del ( netdev , ctx ,
TLS_OFFLOAD_CTX_DIR_TX ) ;
if ( ctx - > rx_conf = = TLS_HW )
netdev - > tlsdev_ops - > tls_dev_del ( netdev , ctx ,
TLS_OFFLOAD_CTX_DIR_RX ) ;
2019-06-04 12:00:12 -07:00
WRITE_ONCE ( ctx - > netdev , NULL ) ;
smp_mb__before_atomic ( ) ; /* pairs with test_and_set_bit() */
while ( test_bit ( TLS_RX_SYNC_RUNNING , & ctx - > flags ) )
usleep_range ( 10 , 200 ) ;
2018-04-30 10:16:16 +03:00
dev_put ( netdev ) ;
list_del_init ( & ctx - > list ) ;
if ( refcount_dec_and_test ( & ctx - > refcount ) )
tls_device_free_ctx ( ctx ) ;
}
up_write ( & device_offload_lock ) ;
flush_work ( & tls_device_gc_work ) ;
return NOTIFY_DONE ;
}
static int tls_dev_event ( struct notifier_block * this , unsigned long event ,
void * ptr )
{
struct net_device * dev = netdev_notifier_info_to_dev ( ptr ) ;
2019-05-21 19:02:02 -07:00
if ( ! dev - > tlsdev_ops & &
! ( dev - > features & ( NETIF_F_HW_TLS_RX | NETIF_F_HW_TLS_TX ) ) )
2018-04-30 10:16:16 +03:00
return NOTIFY_DONE ;
switch ( event ) {
case NETDEV_REGISTER :
case NETDEV_FEAT_CHANGE :
2018-07-13 14:33:43 +03:00
if ( ( dev - > features & NETIF_F_HW_TLS_RX ) & &
2019-06-10 21:40:08 -07:00
! dev - > tlsdev_ops - > tls_dev_resync )
2018-07-13 14:33:43 +03:00
return NOTIFY_BAD ;
2018-04-30 10:16:16 +03:00
if ( dev - > tlsdev_ops & &
dev - > tlsdev_ops - > tls_dev_add & &
dev - > tlsdev_ops - > tls_dev_del )
return NOTIFY_DONE ;
else
return NOTIFY_BAD ;
case NETDEV_DOWN :
return tls_device_down ( dev ) ;
}
return NOTIFY_DONE ;
}
static struct notifier_block tls_dev_notifier = {
. notifier_call = tls_dev_event ,
} ;
void __init tls_device_init ( void )
{
register_netdevice_notifier ( & tls_dev_notifier ) ;
}
void __exit tls_device_cleanup ( void )
{
unregister_netdevice_notifier ( & tls_dev_notifier ) ;
flush_work ( & tls_device_gc_work ) ;
2019-05-08 16:46:14 -07:00
clean_acked_data_flush ( ) ;
2018-04-30 10:16:16 +03:00
}