18c2ecf20Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0-or-later 28c2ecf20Sopenharmony_ci/* 38c2ecf20Sopenharmony_ci * drivers/net/team/team_mode_roundrobin.c - Round-robin mode for team 48c2ecf20Sopenharmony_ci * Copyright (c) 2011 Jiri Pirko <jpirko@redhat.com> 58c2ecf20Sopenharmony_ci */ 68c2ecf20Sopenharmony_ci 78c2ecf20Sopenharmony_ci#include <linux/kernel.h> 88c2ecf20Sopenharmony_ci#include <linux/types.h> 98c2ecf20Sopenharmony_ci#include <linux/module.h> 108c2ecf20Sopenharmony_ci#include <linux/init.h> 118c2ecf20Sopenharmony_ci#include <linux/errno.h> 128c2ecf20Sopenharmony_ci#include <linux/netdevice.h> 138c2ecf20Sopenharmony_ci#include <linux/if_team.h> 148c2ecf20Sopenharmony_ci 158c2ecf20Sopenharmony_cistruct rr_priv { 168c2ecf20Sopenharmony_ci unsigned int sent_packets; 178c2ecf20Sopenharmony_ci}; 188c2ecf20Sopenharmony_ci 198c2ecf20Sopenharmony_cistatic struct rr_priv *rr_priv(struct team *team) 208c2ecf20Sopenharmony_ci{ 218c2ecf20Sopenharmony_ci return (struct rr_priv *) &team->mode_priv; 228c2ecf20Sopenharmony_ci} 238c2ecf20Sopenharmony_ci 248c2ecf20Sopenharmony_cistatic bool rr_transmit(struct team *team, struct sk_buff *skb) 258c2ecf20Sopenharmony_ci{ 268c2ecf20Sopenharmony_ci struct team_port *port; 278c2ecf20Sopenharmony_ci int port_index; 288c2ecf20Sopenharmony_ci 298c2ecf20Sopenharmony_ci port_index = team_num_to_port_index(team, 308c2ecf20Sopenharmony_ci rr_priv(team)->sent_packets++); 318c2ecf20Sopenharmony_ci port = team_get_port_by_index_rcu(team, port_index); 328c2ecf20Sopenharmony_ci if (unlikely(!port)) 338c2ecf20Sopenharmony_ci goto drop; 348c2ecf20Sopenharmony_ci port = team_get_first_port_txable_rcu(team, port); 358c2ecf20Sopenharmony_ci if (unlikely(!port)) 368c2ecf20Sopenharmony_ci goto drop; 378c2ecf20Sopenharmony_ci if (team_dev_queue_xmit(team, port, skb)) 388c2ecf20Sopenharmony_ci return false; 398c2ecf20Sopenharmony_ci return true; 408c2ecf20Sopenharmony_ci 418c2ecf20Sopenharmony_cidrop: 428c2ecf20Sopenharmony_ci dev_kfree_skb_any(skb); 438c2ecf20Sopenharmony_ci return false; 448c2ecf20Sopenharmony_ci} 458c2ecf20Sopenharmony_ci 468c2ecf20Sopenharmony_cistatic const struct team_mode_ops rr_mode_ops = { 478c2ecf20Sopenharmony_ci .transmit = rr_transmit, 488c2ecf20Sopenharmony_ci .port_enter = team_modeop_port_enter, 498c2ecf20Sopenharmony_ci .port_change_dev_addr = team_modeop_port_change_dev_addr, 508c2ecf20Sopenharmony_ci}; 518c2ecf20Sopenharmony_ci 528c2ecf20Sopenharmony_cistatic const struct team_mode rr_mode = { 538c2ecf20Sopenharmony_ci .kind = "roundrobin", 548c2ecf20Sopenharmony_ci .owner = THIS_MODULE, 558c2ecf20Sopenharmony_ci .priv_size = sizeof(struct rr_priv), 568c2ecf20Sopenharmony_ci .ops = &rr_mode_ops, 578c2ecf20Sopenharmony_ci .lag_tx_type = NETDEV_LAG_TX_TYPE_ROUNDROBIN, 588c2ecf20Sopenharmony_ci}; 598c2ecf20Sopenharmony_ci 608c2ecf20Sopenharmony_cistatic int __init rr_init_module(void) 618c2ecf20Sopenharmony_ci{ 628c2ecf20Sopenharmony_ci return team_mode_register(&rr_mode); 638c2ecf20Sopenharmony_ci} 648c2ecf20Sopenharmony_ci 658c2ecf20Sopenharmony_cistatic void __exit rr_cleanup_module(void) 668c2ecf20Sopenharmony_ci{ 678c2ecf20Sopenharmony_ci team_mode_unregister(&rr_mode); 688c2ecf20Sopenharmony_ci} 698c2ecf20Sopenharmony_ci 708c2ecf20Sopenharmony_cimodule_init(rr_init_module); 718c2ecf20Sopenharmony_cimodule_exit(rr_cleanup_module); 728c2ecf20Sopenharmony_ci 738c2ecf20Sopenharmony_ciMODULE_LICENSE("GPL v2"); 748c2ecf20Sopenharmony_ciMODULE_AUTHOR("Jiri Pirko <jpirko@redhat.com>"); 758c2ecf20Sopenharmony_ciMODULE_DESCRIPTION("Round-robin mode for team"); 768c2ecf20Sopenharmony_ciMODULE_ALIAS_TEAM_MODE("roundrobin"); 77