2023-08-21 15:25:14 -07:00
// SPDX-License-Identifier: GPL-2.0
/* Multipath TCP
*
* Copyright ( c ) 2022 , SUSE .
*/
# define pr_fmt(fmt) "MPTCP: " fmt
# include <linux/kernel.h>
# include <linux/module.h>
# include <linux/list.h>
# include <linux/rculist.h>
# include <linux/spinlock.h>
# include "protocol.h"
static DEFINE_SPINLOCK ( mptcp_sched_list_lock ) ;
static LIST_HEAD ( mptcp_sched_list ) ;
2023-08-21 15:25:21 -07:00
static int mptcp_sched_default_get_subflow ( struct mptcp_sock * msk ,
struct mptcp_sched_data * data )
{
struct sock * ssk ;
ssk = data - > reinject ? mptcp_subflow_get_retrans ( msk ) :
mptcp_subflow_get_send ( msk ) ;
if ( ! ssk )
return - EINVAL ;
mptcp_subflow_set_scheduled ( mptcp_subflow_ctx ( ssk ) , true ) ;
return 0 ;
}
static struct mptcp_sched_ops mptcp_sched_default = {
. get_subflow = mptcp_sched_default_get_subflow ,
. name = " default " ,
. owner = THIS_MODULE ,
} ;
2023-08-21 15:25:14 -07:00
/* Must be called with rcu read lock held */
struct mptcp_sched_ops * mptcp_sched_find ( const char * name )
{
struct mptcp_sched_ops * sched , * ret = NULL ;
list_for_each_entry_rcu ( sched , & mptcp_sched_list , list ) {
if ( ! strcmp ( sched - > name , name ) ) {
ret = sched ;
break ;
}
}
return ret ;
}
int mptcp_register_scheduler ( struct mptcp_sched_ops * sched )
{
if ( ! sched - > get_subflow )
return - EINVAL ;
spin_lock ( & mptcp_sched_list_lock ) ;
if ( mptcp_sched_find ( sched - > name ) ) {
spin_unlock ( & mptcp_sched_list_lock ) ;
return - EEXIST ;
}
list_add_tail_rcu ( & sched - > list , & mptcp_sched_list ) ;
spin_unlock ( & mptcp_sched_list_lock ) ;
pr_debug ( " %s registered " , sched - > name ) ;
return 0 ;
}
void mptcp_unregister_scheduler ( struct mptcp_sched_ops * sched )
{
2023-08-21 15:25:21 -07:00
if ( sched = = & mptcp_sched_default )
return ;
2023-08-21 15:25:14 -07:00
spin_lock ( & mptcp_sched_list_lock ) ;
list_del_rcu ( & sched - > list ) ;
spin_unlock ( & mptcp_sched_list_lock ) ;
}
2023-08-21 15:25:16 -07:00
2023-08-21 15:25:21 -07:00
void mptcp_sched_init ( void )
{
mptcp_register_scheduler ( & mptcp_sched_default ) ;
}
2023-08-21 15:25:16 -07:00
int mptcp_init_sched ( struct mptcp_sock * msk ,
struct mptcp_sched_ops * sched )
{
if ( ! sched )
2023-08-21 15:25:21 -07:00
sched = & mptcp_sched_default ;
2023-08-21 15:25:16 -07:00
if ( ! bpf_try_module_get ( sched , sched - > owner ) )
return - EBUSY ;
msk - > sched = sched ;
if ( msk - > sched - > init )
msk - > sched - > init ( msk ) ;
pr_debug ( " sched=%s " , msk - > sched - > name ) ;
return 0 ;
}
void mptcp_release_sched ( struct mptcp_sock * msk )
{
struct mptcp_sched_ops * sched = msk - > sched ;
if ( ! sched )
return ;
msk - > sched = NULL ;
if ( sched - > release )
sched - > release ( msk ) ;
bpf_module_put ( sched , sched - > owner ) ;
}
2023-08-21 15:25:17 -07:00
void mptcp_subflow_set_scheduled ( struct mptcp_subflow_context * subflow ,
bool scheduled )
{
WRITE_ONCE ( subflow - > scheduled , scheduled ) ;
}
2023-08-21 15:25:18 -07:00
int mptcp_sched_get_send ( struct mptcp_sock * msk )
{
struct mptcp_subflow_context * subflow ;
struct mptcp_sched_data data ;
2023-08-21 15:25:19 -07:00
msk_owned_by_me ( msk ) ;
/* the following check is moved out of mptcp_subflow_get_send */
if ( __mptcp_check_fallback ( msk ) ) {
if ( msk - > first & &
__tcp_can_send ( msk - > first ) & &
sk_stream_memory_free ( msk - > first ) ) {
mptcp_subflow_set_scheduled ( mptcp_subflow_ctx ( msk - > first ) , true ) ;
return 0 ;
}
return - EINVAL ;
}
2023-08-21 15:25:18 -07:00
mptcp_for_each_subflow ( msk , subflow ) {
if ( READ_ONCE ( subflow - > scheduled ) )
return 0 ;
}
data . reinject = false ;
2023-08-21 15:25:21 -07:00
if ( msk - > sched = = & mptcp_sched_default | | ! msk - > sched )
return mptcp_sched_default_get_subflow ( msk , & data ) ;
2023-08-21 15:25:18 -07:00
return msk - > sched - > get_subflow ( msk , & data ) ;
}
int mptcp_sched_get_retrans ( struct mptcp_sock * msk )
{
struct mptcp_subflow_context * subflow ;
struct mptcp_sched_data data ;
2023-08-21 15:25:20 -07:00
msk_owned_by_me ( msk ) ;
/* the following check is moved out of mptcp_subflow_get_retrans */
if ( __mptcp_check_fallback ( msk ) )
return - EINVAL ;
2023-08-21 15:25:18 -07:00
mptcp_for_each_subflow ( msk , subflow ) {
if ( READ_ONCE ( subflow - > scheduled ) )
return 0 ;
}
data . reinject = true ;
2023-08-21 15:25:21 -07:00
if ( msk - > sched = = & mptcp_sched_default | | ! msk - > sched )
return mptcp_sched_default_get_subflow ( msk , & data ) ;
2023-08-21 15:25:18 -07:00
return msk - > sched - > get_subflow ( msk , & data ) ;
}