2020-10-27 10:02:45 +00:00
// SPDX-License-Identifier: GPL-2.0-or-later
# include <linux/cfm_bridge.h>
# include <uapi/linux/cfm_bridge.h>
# include "br_private_cfm.h"
static struct br_cfm_mep * br_mep_find ( struct net_bridge * br , u32 instance )
{
struct br_cfm_mep * mep ;
hlist_for_each_entry ( mep , & br - > mep_list , head )
if ( mep - > instance = = instance )
return mep ;
return NULL ;
}
static struct br_cfm_mep * br_mep_find_ifindex ( struct net_bridge * br ,
u32 ifindex )
{
struct br_cfm_mep * mep ;
hlist_for_each_entry_rcu ( mep , & br - > mep_list , head ,
lockdep_rtnl_is_held ( ) )
if ( mep - > create . ifindex = = ifindex )
return mep ;
return NULL ;
}
static struct br_cfm_peer_mep * br_peer_mep_find ( struct br_cfm_mep * mep ,
u32 mepid )
{
struct br_cfm_peer_mep * peer_mep ;
hlist_for_each_entry_rcu ( peer_mep , & mep - > peer_mep_list , head ,
lockdep_rtnl_is_held ( ) )
if ( peer_mep - > mepid = = mepid )
return peer_mep ;
return NULL ;
}
static struct net_bridge_port * br_mep_get_port ( struct net_bridge * br ,
u32 ifindex )
{
struct net_bridge_port * port ;
list_for_each_entry ( port , & br - > port_list , list )
if ( port - > dev - > ifindex = = ifindex )
return port ;
return NULL ;
}
2020-10-27 10:02:46 +00:00
/* Calculate the CCM interval in us. */
static u32 interval_to_us ( enum br_cfm_ccm_interval interval )
{
switch ( interval ) {
case BR_CFM_CCM_INTERVAL_NONE :
return 0 ;
case BR_CFM_CCM_INTERVAL_3_3_MS :
return 3300 ;
case BR_CFM_CCM_INTERVAL_10_MS :
return 10 * 1000 ;
case BR_CFM_CCM_INTERVAL_100_MS :
return 100 * 1000 ;
case BR_CFM_CCM_INTERVAL_1_SEC :
return 1000 * 1000 ;
case BR_CFM_CCM_INTERVAL_10_SEC :
return 10 * 1000 * 1000 ;
case BR_CFM_CCM_INTERVAL_1_MIN :
return 60 * 1000 * 1000 ;
case BR_CFM_CCM_INTERVAL_10_MIN :
return 10 * 60 * 1000 * 1000 ;
}
return 0 ;
}
/* Convert the interface interval to CCM PDU value. */
static u32 interval_to_pdu ( enum br_cfm_ccm_interval interval )
{
switch ( interval ) {
case BR_CFM_CCM_INTERVAL_NONE :
return 0 ;
case BR_CFM_CCM_INTERVAL_3_3_MS :
return 1 ;
case BR_CFM_CCM_INTERVAL_10_MS :
return 2 ;
case BR_CFM_CCM_INTERVAL_100_MS :
return 3 ;
case BR_CFM_CCM_INTERVAL_1_SEC :
return 4 ;
case BR_CFM_CCM_INTERVAL_10_SEC :
return 5 ;
case BR_CFM_CCM_INTERVAL_1_MIN :
return 6 ;
case BR_CFM_CCM_INTERVAL_10_MIN :
return 7 ;
}
return 0 ;
}
2020-10-27 10:02:47 +00:00
/* Convert the CCM PDU value to interval on interface. */
static u32 pdu_to_interval ( u32 value )
{
switch ( value ) {
case 0 :
return BR_CFM_CCM_INTERVAL_NONE ;
case 1 :
return BR_CFM_CCM_INTERVAL_3_3_MS ;
case 2 :
return BR_CFM_CCM_INTERVAL_10_MS ;
case 3 :
return BR_CFM_CCM_INTERVAL_100_MS ;
case 4 :
return BR_CFM_CCM_INTERVAL_1_SEC ;
case 5 :
return BR_CFM_CCM_INTERVAL_10_SEC ;
case 6 :
return BR_CFM_CCM_INTERVAL_1_MIN ;
case 7 :
return BR_CFM_CCM_INTERVAL_10_MIN ;
}
return BR_CFM_CCM_INTERVAL_NONE ;
}
static void ccm_rx_timer_start ( struct br_cfm_peer_mep * peer_mep )
{
u32 interval_us ;
interval_us = interval_to_us ( peer_mep - > mep - > cc_config . exp_interval ) ;
/* Function ccm_rx_dwork must be called with 1/4
* of the configured CC ' expected_interval '
* in order to detect CCM defect after 3.25 interval .
*/
queue_delayed_work ( system_wq , & peer_mep - > ccm_rx_dwork ,
usecs_to_jiffies ( interval_us / 4 ) ) ;
}
2020-10-27 10:02:51 +00:00
static void br_cfm_notify ( int event , const struct net_bridge_port * port )
{
u32 filter = RTEXT_FILTER_CFM_STATUS ;
2021-06-21 23:05:19 -07:00
br_info_notify ( event , port - > br , NULL , filter ) ;
2020-10-27 10:02:51 +00:00
}
2020-10-27 10:02:47 +00:00
static void cc_peer_enable ( struct br_cfm_peer_mep * peer_mep )
{
memset ( & peer_mep - > cc_status , 0 , sizeof ( peer_mep - > cc_status ) ) ;
peer_mep - > ccm_rx_count_miss = 0 ;
ccm_rx_timer_start ( peer_mep ) ;
}
static void cc_peer_disable ( struct br_cfm_peer_mep * peer_mep )
{
cancel_delayed_work_sync ( & peer_mep - > ccm_rx_dwork ) ;
}
2020-10-27 10:02:46 +00:00
static struct sk_buff * ccm_frame_build ( struct br_cfm_mep * mep ,
const struct br_cfm_cc_ccm_tx_info * const tx_info )
{
struct br_cfm_common_hdr * common_hdr ;
struct net_bridge_port * b_port ;
struct br_cfm_maid * maid ;
u8 * itu_reserved , * e_tlv ;
struct ethhdr * eth_hdr ;
struct sk_buff * skb ;
__be32 * status_tlv ;
__be32 * snumber ;
__be16 * mepid ;
skb = dev_alloc_skb ( CFM_CCM_MAX_FRAME_LENGTH ) ;
if ( ! skb )
return NULL ;
rcu_read_lock ( ) ;
b_port = rcu_dereference ( mep - > b_port ) ;
if ( ! b_port ) {
kfree_skb ( skb ) ;
rcu_read_unlock ( ) ;
return NULL ;
}
skb - > dev = b_port - > dev ;
rcu_read_unlock ( ) ;
/* The device cannot be deleted until the work_queue functions has
* completed . This function is called from ccm_tx_work_expired ( )
* that is a work_queue functions .
*/
skb - > protocol = htons ( ETH_P_CFM ) ;
skb - > priority = CFM_FRAME_PRIO ;
/* Ethernet header */
eth_hdr = skb_put ( skb , sizeof ( * eth_hdr ) ) ;
ether_addr_copy ( eth_hdr - > h_dest , tx_info - > dmac . addr ) ;
ether_addr_copy ( eth_hdr - > h_source , mep - > config . unicast_mac . addr ) ;
eth_hdr - > h_proto = htons ( ETH_P_CFM ) ;
/* Common CFM Header */
common_hdr = skb_put ( skb , sizeof ( * common_hdr ) ) ;
common_hdr - > mdlevel_version = mep - > config . mdlevel < < 5 ;
common_hdr - > opcode = BR_CFM_OPCODE_CCM ;
common_hdr - > flags = ( mep - > rdi < < 7 ) |
interval_to_pdu ( mep - > cc_config . exp_interval ) ;
common_hdr - > tlv_offset = CFM_CCM_TLV_OFFSET ;
/* Sequence number */
snumber = skb_put ( skb , sizeof ( * snumber ) ) ;
if ( tx_info - > seq_no_update ) {
* snumber = cpu_to_be32 ( mep - > ccm_tx_snumber ) ;
mep - > ccm_tx_snumber + = 1 ;
} else {
* snumber = 0 ;
}
mepid = skb_put ( skb , sizeof ( * mepid ) ) ;
* mepid = cpu_to_be16 ( ( u16 ) mep - > config . mepid ) ;
maid = skb_put ( skb , sizeof ( * maid ) ) ;
memcpy ( maid - > data , mep - > cc_config . exp_maid . data , sizeof ( maid - > data ) ) ;
/* ITU reserved (CFM_CCM_ITU_RESERVED_SIZE octets) */
itu_reserved = skb_put ( skb , CFM_CCM_ITU_RESERVED_SIZE ) ;
memset ( itu_reserved , 0 , CFM_CCM_ITU_RESERVED_SIZE ) ;
/* Generel CFM TLV format:
* TLV type : one byte
* TLV value length : two bytes
* TLV value : ' TLV value length ' bytes
*/
/* Port status TLV. The value length is 1. Total of 4 bytes. */
if ( tx_info - > port_tlv ) {
status_tlv = skb_put ( skb , sizeof ( * status_tlv ) ) ;
* status_tlv = cpu_to_be32 ( ( CFM_PORT_STATUS_TLV_TYPE < < 24 ) |
( 1 < < 8 ) | /* Value length */
( tx_info - > port_tlv_value & 0xFF ) ) ;
}
/* Interface status TLV. The value length is 1. Total of 4 bytes. */
if ( tx_info - > if_tlv ) {
status_tlv = skb_put ( skb , sizeof ( * status_tlv ) ) ;
* status_tlv = cpu_to_be32 ( ( CFM_IF_STATUS_TLV_TYPE < < 24 ) |
( 1 < < 8 ) | /* Value length */
( tx_info - > if_tlv_value & 0xFF ) ) ;
}
/* End TLV */
e_tlv = skb_put ( skb , sizeof ( * e_tlv ) ) ;
* e_tlv = CFM_ENDE_TLV_TYPE ;
return skb ;
}
static void ccm_frame_tx ( struct sk_buff * skb )
{
skb_reset_network_header ( skb ) ;
dev_queue_xmit ( skb ) ;
}
/* This function is called with the configured CC 'expected_interval'
* in order to drive CCM transmission when enabled .
*/
static void ccm_tx_work_expired ( struct work_struct * work )
{
struct delayed_work * del_work ;
struct br_cfm_mep * mep ;
struct sk_buff * skb ;
u32 interval_us ;
del_work = to_delayed_work ( work ) ;
mep = container_of ( del_work , struct br_cfm_mep , ccm_tx_dwork ) ;
if ( time_before_eq ( mep - > ccm_tx_end , jiffies ) ) {
/* Transmission period has ended */
mep - > cc_ccm_tx_info . period = 0 ;
return ;
}
skb = ccm_frame_build ( mep , & mep - > cc_ccm_tx_info ) ;
if ( skb )
ccm_frame_tx ( skb ) ;
interval_us = interval_to_us ( mep - > cc_config . exp_interval ) ;
queue_delayed_work ( system_wq , & mep - > ccm_tx_dwork ,
usecs_to_jiffies ( interval_us ) ) ;
}
2020-10-27 10:02:47 +00:00
/* This function is called with 1/4 of the configured CC 'expected_interval'
* in order to detect CCM defect after 3.25 interval .
*/
static void ccm_rx_work_expired ( struct work_struct * work )
{
struct br_cfm_peer_mep * peer_mep ;
2020-10-27 10:02:51 +00:00
struct net_bridge_port * b_port ;
2020-10-27 10:02:47 +00:00
struct delayed_work * del_work ;
del_work = to_delayed_work ( work ) ;
peer_mep = container_of ( del_work , struct br_cfm_peer_mep , ccm_rx_dwork ) ;
/* After 13 counts (4 * 3,25) then 3.25 intervals are expired */
if ( peer_mep - > ccm_rx_count_miss < 13 ) {
/* 3.25 intervals are NOT expired without CCM reception */
peer_mep - > ccm_rx_count_miss + + ;
/* Start timer again */
ccm_rx_timer_start ( peer_mep ) ;
} else {
/* 3.25 intervals are expired without CCM reception.
* CCM defect detected
*/
peer_mep - > cc_status . ccm_defect = true ;
2020-10-27 10:02:51 +00:00
/* Change in CCM defect status - notify */
rcu_read_lock ( ) ;
b_port = rcu_dereference ( peer_mep - > mep - > b_port ) ;
if ( b_port )
br_cfm_notify ( RTM_NEWLINK , b_port ) ;
rcu_read_unlock ( ) ;
2020-10-27 10:02:47 +00:00
}
}
static u32 ccm_tlv_extract ( struct sk_buff * skb , u32 index ,
struct br_cfm_peer_mep * peer_mep )
{
__be32 * s_tlv ;
__be32 _s_tlv ;
u32 h_s_tlv ;
u8 * e_tlv ;
u8 _e_tlv ;
e_tlv = skb_header_pointer ( skb , index , sizeof ( _e_tlv ) , & _e_tlv ) ;
if ( ! e_tlv )
return 0 ;
/* TLV is present - get the status TLV */
s_tlv = skb_header_pointer ( skb ,
index ,
sizeof ( _s_tlv ) , & _s_tlv ) ;
if ( ! s_tlv )
return 0 ;
h_s_tlv = ntohl ( * s_tlv ) ;
if ( ( h_s_tlv > > 24 ) = = CFM_IF_STATUS_TLV_TYPE ) {
/* Interface status TLV */
peer_mep - > cc_status . tlv_seen = true ;
peer_mep - > cc_status . if_tlv_value = ( h_s_tlv & 0xFF ) ;
}
if ( ( h_s_tlv > > 24 ) = = CFM_PORT_STATUS_TLV_TYPE ) {
/* Port status TLV */
peer_mep - > cc_status . tlv_seen = true ;
peer_mep - > cc_status . port_tlv_value = ( h_s_tlv & 0xFF ) ;
}
/* The Sender ID TLV is not handled */
/* The Organization-Specific TLV is not handled */
/* Return the length of this tlv.
* This is the length of the value field plus 3 bytes for size of type
* field and length field
*/
return ( ( h_s_tlv > > 8 ) & 0xFFFF ) + 3 ;
}
/* note: already called with rcu_read_lock */
static int br_cfm_frame_rx ( struct net_bridge_port * port , struct sk_buff * skb )
{
u32 mdlevel , interval , size , index , max ;
const struct br_cfm_common_hdr * hdr ;
struct br_cfm_peer_mep * peer_mep ;
const struct br_cfm_maid * maid ;
struct br_cfm_common_hdr _hdr ;
struct br_cfm_maid _maid ;
struct br_cfm_mep * mep ;
struct net_bridge * br ;
__be32 * snumber ;
__be32 _snumber ;
__be16 * mepid ;
__be16 _mepid ;
if ( port - > state = = BR_STATE_DISABLED )
return 0 ;
hdr = skb_header_pointer ( skb , 0 , sizeof ( _hdr ) , & _hdr ) ;
if ( ! hdr )
return 1 ;
br = port - > br ;
mep = br_mep_find_ifindex ( br , port - > dev - > ifindex ) ;
if ( unlikely ( ! mep ) )
/* No MEP on this port - must be forwarded */
return 0 ;
mdlevel = hdr - > mdlevel_version > > 5 ;
if ( mdlevel > mep - > config . mdlevel )
/* The level is above this MEP level - must be forwarded */
return 0 ;
if ( ( hdr - > mdlevel_version & 0x1F ) ! = 0 ) {
/* Invalid version */
mep - > status . version_unexp_seen = true ;
return 1 ;
}
if ( mdlevel < mep - > config . mdlevel ) {
/* The level is below this MEP level */
mep - > status . rx_level_low_seen = true ;
return 1 ;
}
if ( hdr - > opcode = = BR_CFM_OPCODE_CCM ) {
/* CCM PDU received. */
/* MA ID is after common header + sequence number + MEP ID */
maid = skb_header_pointer ( skb ,
CFM_CCM_PDU_MAID_OFFSET ,
sizeof ( _maid ) , & _maid ) ;
if ( ! maid )
return 1 ;
if ( memcmp ( maid - > data , mep - > cc_config . exp_maid . data ,
sizeof ( maid - > data ) ) )
/* MA ID not as expected */
return 1 ;
/* MEP ID is after common header + sequence number */
mepid = skb_header_pointer ( skb ,
CFM_CCM_PDU_MEPID_OFFSET ,
sizeof ( _mepid ) , & _mepid ) ;
if ( ! mepid )
return 1 ;
peer_mep = br_peer_mep_find ( mep , ( u32 ) ntohs ( * mepid ) ) ;
if ( ! peer_mep )
return 1 ;
/* Interval is in common header flags */
interval = hdr - > flags & 0x07 ;
if ( mep - > cc_config . exp_interval ! = pdu_to_interval ( interval ) )
/* Interval not as expected */
return 1 ;
/* A valid CCM frame is received */
if ( peer_mep - > cc_status . ccm_defect ) {
peer_mep - > cc_status . ccm_defect = false ;
2020-10-27 10:02:51 +00:00
/* Change in CCM defect status - notify */
br_cfm_notify ( RTM_NEWLINK , port ) ;
2020-10-27 10:02:47 +00:00
/* Start CCM RX timer */
ccm_rx_timer_start ( peer_mep ) ;
}
peer_mep - > cc_status . seen = true ;
peer_mep - > ccm_rx_count_miss = 0 ;
/* RDI is in common header flags */
peer_mep - > cc_status . rdi = ( hdr - > flags & 0x80 ) ? true : false ;
/* Sequence number is after common header */
snumber = skb_header_pointer ( skb ,
CFM_CCM_PDU_SEQNR_OFFSET ,
sizeof ( _snumber ) , & _snumber ) ;
if ( ! snumber )
return 1 ;
if ( ntohl ( * snumber ) ! = ( mep - > ccm_rx_snumber + 1 ) )
/* Unexpected sequence number */
peer_mep - > cc_status . seq_unexp_seen = true ;
mep - > ccm_rx_snumber = ntohl ( * snumber ) ;
/* TLV end is after common header + sequence number + MEP ID +
* MA ID + ITU reserved
*/
index = CFM_CCM_PDU_TLV_OFFSET ;
max = 0 ;
do { /* Handle all TLVs */
size = ccm_tlv_extract ( skb , index , peer_mep ) ;
index + = size ;
max + = 1 ;
} while ( size ! = 0 & & max < 4 ) ; /* Max four TLVs possible */
return 1 ;
}
mep - > status . opcode_unexp_seen = true ;
return 1 ;
}
static struct br_frame_type cfm_frame_type __read_mostly = {
. type = cpu_to_be16 ( ETH_P_CFM ) ,
. frame_handler = br_cfm_frame_rx ,
} ;
2020-10-27 10:02:45 +00:00
int br_cfm_mep_create ( struct net_bridge * br ,
const u32 instance ,
struct br_cfm_mep_create * const create ,
struct netlink_ext_ack * extack )
{
struct net_bridge_port * p ;
struct br_cfm_mep * mep ;
ASSERT_RTNL ( ) ;
if ( create - > domain = = BR_CFM_VLAN ) {
NL_SET_ERR_MSG_MOD ( extack ,
" VLAN domain not supported " ) ;
return - EINVAL ;
}
if ( create - > domain ! = BR_CFM_PORT ) {
NL_SET_ERR_MSG_MOD ( extack ,
" Invalid domain value " ) ;
return - EINVAL ;
}
if ( create - > direction = = BR_CFM_MEP_DIRECTION_UP ) {
NL_SET_ERR_MSG_MOD ( extack ,
" Up-MEP not supported " ) ;
return - EINVAL ;
}
if ( create - > direction ! = BR_CFM_MEP_DIRECTION_DOWN ) {
NL_SET_ERR_MSG_MOD ( extack ,
" Invalid direction value " ) ;
return - EINVAL ;
}
p = br_mep_get_port ( br , create - > ifindex ) ;
if ( ! p ) {
NL_SET_ERR_MSG_MOD ( extack ,
" Port is not related to bridge " ) ;
return - EINVAL ;
}
mep = br_mep_find ( br , instance ) ;
if ( mep ) {
NL_SET_ERR_MSG_MOD ( extack ,
" MEP instance already exists " ) ;
return - EEXIST ;
}
/* In PORT domain only one instance can be created per port */
if ( create - > domain = = BR_CFM_PORT ) {
mep = br_mep_find_ifindex ( br , create - > ifindex ) ;
if ( mep ) {
NL_SET_ERR_MSG_MOD ( extack ,
" Only one Port MEP on a port allowed " ) ;
return - EINVAL ;
}
}
mep = kzalloc ( sizeof ( * mep ) , GFP_KERNEL ) ;
if ( ! mep )
return - ENOMEM ;
mep - > create = * create ;
mep - > instance = instance ;
rcu_assign_pointer ( mep - > b_port , p ) ;
INIT_HLIST_HEAD ( & mep - > peer_mep_list ) ;
2020-10-27 10:02:46 +00:00
INIT_DELAYED_WORK ( & mep - > ccm_tx_dwork , ccm_tx_work_expired ) ;
2020-10-27 10:02:45 +00:00
2020-10-27 10:02:47 +00:00
if ( hlist_empty ( & br - > mep_list ) )
br_add_frame ( br , & cfm_frame_type ) ;
2020-10-27 10:02:45 +00:00
hlist_add_tail_rcu ( & mep - > head , & br - > mep_list ) ;
return 0 ;
}
static void mep_delete_implementation ( struct net_bridge * br ,
struct br_cfm_mep * mep )
{
struct br_cfm_peer_mep * peer_mep ;
struct hlist_node * n_store ;
ASSERT_RTNL ( ) ;
/* Empty and free peer MEP list */
hlist_for_each_entry_safe ( peer_mep , n_store , & mep - > peer_mep_list , head ) {
2020-10-27 10:02:47 +00:00
cancel_delayed_work_sync ( & peer_mep - > ccm_rx_dwork ) ;
2020-10-27 10:02:45 +00:00
hlist_del_rcu ( & peer_mep - > head ) ;
kfree_rcu ( peer_mep , rcu ) ;
}
2020-10-27 10:02:46 +00:00
cancel_delayed_work_sync ( & mep - > ccm_tx_dwork ) ;
2020-10-27 10:02:45 +00:00
RCU_INIT_POINTER ( mep - > b_port , NULL ) ;
hlist_del_rcu ( & mep - > head ) ;
kfree_rcu ( mep , rcu ) ;
2020-10-27 10:02:47 +00:00
if ( hlist_empty ( & br - > mep_list ) )
br_del_frame ( br , & cfm_frame_type ) ;
2020-10-27 10:02:45 +00:00
}
int br_cfm_mep_delete ( struct net_bridge * br ,
const u32 instance ,
struct netlink_ext_ack * extack )
{
struct br_cfm_mep * mep ;
ASSERT_RTNL ( ) ;
mep = br_mep_find ( br , instance ) ;
if ( ! mep ) {
NL_SET_ERR_MSG_MOD ( extack ,
" MEP instance does not exists " ) ;
return - ENOENT ;
}
mep_delete_implementation ( br , mep ) ;
return 0 ;
}
int br_cfm_mep_config_set ( struct net_bridge * br ,
const u32 instance ,
const struct br_cfm_mep_config * const config ,
struct netlink_ext_ack * extack )
{
struct br_cfm_mep * mep ;
ASSERT_RTNL ( ) ;
mep = br_mep_find ( br , instance ) ;
if ( ! mep ) {
NL_SET_ERR_MSG_MOD ( extack ,
" MEP instance does not exists " ) ;
return - ENOENT ;
}
mep - > config = * config ;
return 0 ;
}
2020-10-27 10:02:46 +00:00
int br_cfm_cc_config_set ( struct net_bridge * br ,
const u32 instance ,
const struct br_cfm_cc_config * const config ,
struct netlink_ext_ack * extack )
{
2020-10-27 10:02:47 +00:00
struct br_cfm_peer_mep * peer_mep ;
2020-10-27 10:02:46 +00:00
struct br_cfm_mep * mep ;
ASSERT_RTNL ( ) ;
mep = br_mep_find ( br , instance ) ;
if ( ! mep ) {
NL_SET_ERR_MSG_MOD ( extack ,
" MEP instance does not exists " ) ;
return - ENOENT ;
}
/* Check for no change in configuration */
if ( memcmp ( config , & mep - > cc_config , sizeof ( * config ) ) = = 0 )
return 0 ;
2020-10-27 10:02:47 +00:00
if ( config - > enable & & ! mep - > cc_config . enable )
/* CC is enabled */
hlist_for_each_entry ( peer_mep , & mep - > peer_mep_list , head )
cc_peer_enable ( peer_mep ) ;
if ( ! config - > enable & & mep - > cc_config . enable )
/* CC is disabled */
hlist_for_each_entry ( peer_mep , & mep - > peer_mep_list , head )
cc_peer_disable ( peer_mep ) ;
2020-10-27 10:02:46 +00:00
mep - > cc_config = * config ;
2020-10-27 10:02:47 +00:00
mep - > ccm_rx_snumber = 0 ;
2020-10-27 10:02:46 +00:00
mep - > ccm_tx_snumber = 1 ;
return 0 ;
}
2020-10-27 10:02:45 +00:00
int br_cfm_cc_peer_mep_add ( struct net_bridge * br , const u32 instance ,
u32 mepid ,
struct netlink_ext_ack * extack )
{
struct br_cfm_peer_mep * peer_mep ;
struct br_cfm_mep * mep ;
ASSERT_RTNL ( ) ;
mep = br_mep_find ( br , instance ) ;
if ( ! mep ) {
NL_SET_ERR_MSG_MOD ( extack ,
" MEP instance does not exists " ) ;
return - ENOENT ;
}
peer_mep = br_peer_mep_find ( mep , mepid ) ;
if ( peer_mep ) {
NL_SET_ERR_MSG_MOD ( extack ,
" Peer MEP-ID already exists " ) ;
return - EEXIST ;
}
peer_mep = kzalloc ( sizeof ( * peer_mep ) , GFP_KERNEL ) ;
if ( ! peer_mep )
return - ENOMEM ;
peer_mep - > mepid = mepid ;
peer_mep - > mep = mep ;
2020-10-27 10:02:47 +00:00
INIT_DELAYED_WORK ( & peer_mep - > ccm_rx_dwork , ccm_rx_work_expired ) ;
if ( mep - > cc_config . enable )
cc_peer_enable ( peer_mep ) ;
2020-10-27 10:02:45 +00:00
hlist_add_tail_rcu ( & peer_mep - > head , & mep - > peer_mep_list ) ;
return 0 ;
}
int br_cfm_cc_peer_mep_remove ( struct net_bridge * br , const u32 instance ,
u32 mepid ,
struct netlink_ext_ack * extack )
{
struct br_cfm_peer_mep * peer_mep ;
struct br_cfm_mep * mep ;
ASSERT_RTNL ( ) ;
mep = br_mep_find ( br , instance ) ;
if ( ! mep ) {
NL_SET_ERR_MSG_MOD ( extack ,
" MEP instance does not exists " ) ;
return - ENOENT ;
}
peer_mep = br_peer_mep_find ( mep , mepid ) ;
if ( ! peer_mep ) {
NL_SET_ERR_MSG_MOD ( extack ,
" Peer MEP-ID does not exists " ) ;
return - ENOENT ;
}
2020-10-27 10:02:47 +00:00
cc_peer_disable ( peer_mep ) ;
2020-10-27 10:02:45 +00:00
hlist_del_rcu ( & peer_mep - > head ) ;
kfree_rcu ( peer_mep , rcu ) ;
return 0 ;
}
2020-10-27 10:02:46 +00:00
int br_cfm_cc_rdi_set ( struct net_bridge * br , const u32 instance ,
const bool rdi , struct netlink_ext_ack * extack )
{
struct br_cfm_mep * mep ;
ASSERT_RTNL ( ) ;
mep = br_mep_find ( br , instance ) ;
if ( ! mep ) {
NL_SET_ERR_MSG_MOD ( extack ,
" MEP instance does not exists " ) ;
return - ENOENT ;
}
mep - > rdi = rdi ;
return 0 ;
}
int br_cfm_cc_ccm_tx ( struct net_bridge * br , const u32 instance ,
const struct br_cfm_cc_ccm_tx_info * const tx_info ,
struct netlink_ext_ack * extack )
{
struct br_cfm_mep * mep ;
ASSERT_RTNL ( ) ;
mep = br_mep_find ( br , instance ) ;
if ( ! mep ) {
NL_SET_ERR_MSG_MOD ( extack ,
" MEP instance does not exists " ) ;
return - ENOENT ;
}
if ( memcmp ( tx_info , & mep - > cc_ccm_tx_info , sizeof ( * tx_info ) ) = = 0 ) {
/* No change in tx_info. */
if ( mep - > cc_ccm_tx_info . period = = 0 )
/* Transmission is not enabled - just return */
return 0 ;
/* Transmission is ongoing, the end time is recalculated */
mep - > ccm_tx_end = jiffies +
usecs_to_jiffies ( tx_info - > period * 1000000 ) ;
return 0 ;
}
if ( tx_info - > period = = 0 & & mep - > cc_ccm_tx_info . period = = 0 )
/* Some change in info and transmission is not ongoing */
goto save ;
if ( tx_info - > period ! = 0 & & mep - > cc_ccm_tx_info . period ! = 0 ) {
/* Some change in info and transmission is ongoing
* The end time is recalculated
*/
mep - > ccm_tx_end = jiffies +
usecs_to_jiffies ( tx_info - > period * 1000000 ) ;
goto save ;
}
if ( tx_info - > period = = 0 & & mep - > cc_ccm_tx_info . period ! = 0 ) {
cancel_delayed_work_sync ( & mep - > ccm_tx_dwork ) ;
goto save ;
}
/* Start delayed work to transmit CCM frames. It is done with zero delay
* to send first frame immediately
*/
mep - > ccm_tx_end = jiffies + usecs_to_jiffies ( tx_info - > period * 1000000 ) ;
queue_delayed_work ( system_wq , & mep - > ccm_tx_dwork , 0 ) ;
save :
mep - > cc_ccm_tx_info = * tx_info ;
return 0 ;
}
2020-10-27 10:02:51 +00:00
int br_cfm_mep_count ( struct net_bridge * br , u32 * count )
{
struct br_cfm_mep * mep ;
* count = 0 ;
rcu_read_lock ( ) ;
hlist_for_each_entry_rcu ( mep , & br - > mep_list , head )
* count + = 1 ;
rcu_read_unlock ( ) ;
return 0 ;
}
int br_cfm_peer_mep_count ( struct net_bridge * br , u32 * count )
{
struct br_cfm_peer_mep * peer_mep ;
struct br_cfm_mep * mep ;
* count = 0 ;
rcu_read_lock ( ) ;
hlist_for_each_entry_rcu ( mep , & br - > mep_list , head )
hlist_for_each_entry_rcu ( peer_mep , & mep - > peer_mep_list , head )
* count + = 1 ;
rcu_read_unlock ( ) ;
return 0 ;
}
2020-10-27 10:02:48 +00:00
bool br_cfm_created ( struct net_bridge * br )
{
return ! hlist_empty ( & br - > mep_list ) ;
}
2020-10-27 10:02:45 +00:00
/* Deletes the CFM instances on a specific bridge port
*/
void br_cfm_port_del ( struct net_bridge * br , struct net_bridge_port * port )
{
struct hlist_node * n_store ;
struct br_cfm_mep * mep ;
ASSERT_RTNL ( ) ;
hlist_for_each_entry_safe ( mep , n_store , & br - > mep_list , head )
if ( mep - > create . ifindex = = port - > dev - > ifindex )
mep_delete_implementation ( br , mep ) ;
}