dc32cbb3db
This is the third commit of the implementation of the CFM protocol according to 802.1Q section 12.14. Functionality is extended with CCM frame reception. The MEP instance now contains CCM based status information. Most important is the CCM defect status indicating if correct CCM frames are received with the expected interval. Signed-off-by: Henrik Bjoernlund <henrik.bjoernlund@microchip.com> Reviewed-by: Horatiu Vultur <horatiu.vultur@microchip.com> Acked-by: Nikolay Aleksandrov <nikolay@nvidia.com> Signed-off-by: Jakub Kicinski <kuba@kernel.org>
148 lines
4.3 KiB
C
148 lines
4.3 KiB
C
/* SPDX-License-Identifier: GPL-2.0-or-later */
|
|
|
|
#ifndef _BR_PRIVATE_CFM_H_
|
|
#define _BR_PRIVATE_CFM_H_
|
|
|
|
#include "br_private.h"
|
|
#include <uapi/linux/cfm_bridge.h>
|
|
|
|
struct br_cfm_mep_create {
|
|
enum br_cfm_domain domain; /* Domain for this MEP */
|
|
enum br_cfm_mep_direction direction; /* Up or Down MEP direction */
|
|
u32 ifindex; /* Residence port */
|
|
};
|
|
|
|
int br_cfm_mep_create(struct net_bridge *br,
|
|
const u32 instance,
|
|
struct br_cfm_mep_create *const create,
|
|
struct netlink_ext_ack *extack);
|
|
|
|
int br_cfm_mep_delete(struct net_bridge *br,
|
|
const u32 instance,
|
|
struct netlink_ext_ack *extack);
|
|
|
|
struct br_cfm_mep_config {
|
|
u32 mdlevel;
|
|
u32 mepid; /* MEPID for this MEP */
|
|
struct mac_addr unicast_mac; /* The MEP unicast MAC */
|
|
};
|
|
|
|
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_maid {
|
|
u8 data[CFM_MAID_LENGTH];
|
|
};
|
|
|
|
struct br_cfm_cc_config {
|
|
/* Expected received CCM PDU MAID. */
|
|
struct br_cfm_maid exp_maid;
|
|
|
|
/* Expected received CCM PDU interval. */
|
|
/* Transmitting CCM PDU interval when CCM tx is enabled. */
|
|
enum br_cfm_ccm_interval exp_interval;
|
|
|
|
bool enable; /* Enable/disable CCM PDU handling */
|
|
};
|
|
|
|
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);
|
|
|
|
int br_cfm_cc_peer_mep_add(struct net_bridge *br, const u32 instance,
|
|
u32 peer_mep_id,
|
|
struct netlink_ext_ack *extack);
|
|
int br_cfm_cc_peer_mep_remove(struct net_bridge *br, const u32 instance,
|
|
u32 peer_mep_id,
|
|
struct netlink_ext_ack *extack);
|
|
|
|
/* Transmitted CCM Remote Defect Indication status set.
|
|
* This RDI is inserted in transmitted CCM PDUs if CCM transmission is enabled.
|
|
* See br_cfm_cc_ccm_tx() with interval != BR_CFM_CCM_INTERVAL_NONE
|
|
*/
|
|
int br_cfm_cc_rdi_set(struct net_bridge *br, const u32 instance,
|
|
const bool rdi, struct netlink_ext_ack *extack);
|
|
|
|
/* OAM PDU Tx information */
|
|
struct br_cfm_cc_ccm_tx_info {
|
|
struct mac_addr dmac;
|
|
/* The CCM will be transmitted for this period in seconds.
|
|
* Call br_cfm_cc_ccm_tx before timeout to keep transmission alive.
|
|
* When period is zero any ongoing transmission will be stopped.
|
|
*/
|
|
u32 period;
|
|
|
|
bool seq_no_update; /* Update Tx CCM sequence number */
|
|
bool if_tlv; /* Insert Interface Status TLV */
|
|
u8 if_tlv_value; /* Interface Status TLV value */
|
|
bool port_tlv; /* Insert Port Status TLV */
|
|
u8 port_tlv_value; /* Port Status TLV value */
|
|
/* Sender ID TLV ??
|
|
* Organization-Specific TLV ??
|
|
*/
|
|
};
|
|
|
|
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_status {
|
|
/* Indications that an OAM PDU has been seen. */
|
|
bool opcode_unexp_seen; /* RX of OAM PDU with unexpected opcode */
|
|
bool version_unexp_seen; /* RX of OAM PDU with unexpected version */
|
|
bool rx_level_low_seen; /* Rx of OAM PDU with level low */
|
|
};
|
|
|
|
struct br_cfm_cc_peer_status {
|
|
/* This CCM related status is based on the latest received CCM PDU. */
|
|
u8 port_tlv_value; /* Port Status TLV value */
|
|
u8 if_tlv_value; /* Interface Status TLV value */
|
|
|
|
/* CCM has not been received for 3.25 intervals */
|
|
u8 ccm_defect:1;
|
|
|
|
/* (RDI == 1) for last received CCM PDU */
|
|
u8 rdi:1;
|
|
|
|
/* Indications that a CCM PDU has been seen. */
|
|
u8 seen:1; /* CCM PDU received */
|
|
u8 tlv_seen:1; /* CCM PDU with TLV received */
|
|
/* CCM PDU with unexpected sequence number received */
|
|
u8 seq_unexp_seen:1;
|
|
};
|
|
|
|
struct br_cfm_mep {
|
|
/* list header of MEP instances */
|
|
struct hlist_node head;
|
|
u32 instance;
|
|
struct br_cfm_mep_create create;
|
|
struct br_cfm_mep_config config;
|
|
struct br_cfm_cc_config cc_config;
|
|
struct br_cfm_cc_ccm_tx_info cc_ccm_tx_info;
|
|
/* List of multiple peer MEPs */
|
|
struct hlist_head peer_mep_list;
|
|
struct net_bridge_port __rcu *b_port;
|
|
unsigned long ccm_tx_end;
|
|
struct delayed_work ccm_tx_dwork;
|
|
u32 ccm_tx_snumber;
|
|
u32 ccm_rx_snumber;
|
|
struct br_cfm_mep_status status;
|
|
bool rdi;
|
|
struct rcu_head rcu;
|
|
};
|
|
|
|
struct br_cfm_peer_mep {
|
|
struct hlist_node head;
|
|
struct br_cfm_mep *mep;
|
|
struct delayed_work ccm_rx_dwork;
|
|
u32 mepid;
|
|
struct br_cfm_cc_peer_status cc_status;
|
|
u32 ccm_rx_count_miss;
|
|
struct rcu_head rcu;
|
|
};
|
|
|
|
#endif /* _BR_PRIVATE_CFM_H_ */
|