2020-10-27 18:02:45 +08:00
|
|
|
/* 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);
|
|
|
|
|
2020-10-27 18:02:46 +08:00
|
|
|
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;
|
2020-10-27 18:02:47 +08:00
|
|
|
|
|
|
|
bool enable; /* Enable/disable CCM PDU handling */
|
2020-10-27 18:02:46 +08: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 18:02:45 +08:00
|
|
|
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);
|
|
|
|
|
2020-10-27 18:02:46 +08:00
|
|
|
/* 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);
|
|
|
|
|
2020-10-27 18:02:47 +08:00
|
|
|
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;
|
|
|
|
};
|
|
|
|
|
2020-10-27 18:02:45 +08:00
|
|
|
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;
|
2020-10-27 18:02:46 +08:00
|
|
|
struct br_cfm_cc_config cc_config;
|
|
|
|
struct br_cfm_cc_ccm_tx_info cc_ccm_tx_info;
|
2020-10-27 18:02:45 +08:00
|
|
|
/* List of multiple peer MEPs */
|
|
|
|
struct hlist_head peer_mep_list;
|
|
|
|
struct net_bridge_port __rcu *b_port;
|
2020-10-27 18:02:46 +08:00
|
|
|
unsigned long ccm_tx_end;
|
|
|
|
struct delayed_work ccm_tx_dwork;
|
|
|
|
u32 ccm_tx_snumber;
|
2020-10-27 18:02:47 +08:00
|
|
|
u32 ccm_rx_snumber;
|
|
|
|
struct br_cfm_mep_status status;
|
2020-10-27 18:02:46 +08:00
|
|
|
bool rdi;
|
2020-10-27 18:02:45 +08:00
|
|
|
struct rcu_head rcu;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct br_cfm_peer_mep {
|
|
|
|
struct hlist_node head;
|
|
|
|
struct br_cfm_mep *mep;
|
2020-10-27 18:02:47 +08:00
|
|
|
struct delayed_work ccm_rx_dwork;
|
2020-10-27 18:02:45 +08:00
|
|
|
u32 mepid;
|
2020-10-27 18:02:47 +08:00
|
|
|
struct br_cfm_cc_peer_status cc_status;
|
|
|
|
u32 ccm_rx_count_miss;
|
2020-10-27 18:02:45 +08:00
|
|
|
struct rcu_head rcu;
|
|
|
|
};
|
|
|
|
|
|
|
|
#endif /* _BR_PRIVATE_CFM_H_ */
|