| /* SPDX-License-Identifier: GPL-2.0 */ | 
 | /* | 
 |  *  Shared Memory Communications over RDMA (SMC-R) and RoCE | 
 |  * | 
 |  *  Definitions for LLC (link layer control) message handling | 
 |  * | 
 |  *  Copyright IBM Corp. 2016 | 
 |  * | 
 |  *  Author(s):  Klaus Wacker <Klaus.Wacker@de.ibm.com> | 
 |  *              Ursula Braun <ubraun@linux.vnet.ibm.com> | 
 |  */ | 
 |  | 
 | #ifndef SMC_LLC_H | 
 | #define SMC_LLC_H | 
 |  | 
 | #include "smc_wr.h" | 
 |  | 
 | #define SMC_LLC_FLAG_RESP		0x80 | 
 |  | 
 | #define SMC_LLC_WAIT_FIRST_TIME		(5 * HZ) | 
 | #define SMC_LLC_WAIT_TIME		(2 * HZ) | 
 | #define SMC_LLC_TESTLINK_DEFAULT_TIME	(30 * HZ) | 
 |  | 
 | enum smc_llc_reqresp { | 
 | 	SMC_LLC_REQ, | 
 | 	SMC_LLC_RESP | 
 | }; | 
 |  | 
 | enum smc_llc_msg_type { | 
 | 	SMC_LLC_CONFIRM_LINK		= 0x01, | 
 | 	SMC_LLC_ADD_LINK		= 0x02, | 
 | 	SMC_LLC_ADD_LINK_CONT		= 0x03, | 
 | 	SMC_LLC_DELETE_LINK		= 0x04, | 
 | 	SMC_LLC_REQ_ADD_LINK		= 0x05, | 
 | 	SMC_LLC_CONFIRM_RKEY		= 0x06, | 
 | 	SMC_LLC_TEST_LINK		= 0x07, | 
 | 	SMC_LLC_CONFIRM_RKEY_CONT	= 0x08, | 
 | 	SMC_LLC_DELETE_RKEY		= 0x09, | 
 | 	/* V2 types */ | 
 | 	SMC_LLC_CONFIRM_LINK_V2		= 0x21, | 
 | 	SMC_LLC_ADD_LINK_V2		= 0x22, | 
 | 	SMC_LLC_DELETE_LINK_V2		= 0x24, | 
 | 	SMC_LLC_REQ_ADD_LINK_V2		= 0x25, | 
 | 	SMC_LLC_CONFIRM_RKEY_V2		= 0x26, | 
 | 	SMC_LLC_TEST_LINK_V2		= 0x27, | 
 | 	SMC_LLC_DELETE_RKEY_V2		= 0x29, | 
 | }; | 
 |  | 
 | #define smc_link_downing(state) \ | 
 | 	(cmpxchg(state, SMC_LNK_ACTIVE, SMC_LNK_INACTIVE) == SMC_LNK_ACTIVE) | 
 |  | 
 | /* LLC DELETE LINK Request Reason Codes */ | 
 | #define SMC_LLC_DEL_LOST_PATH		0x00010000 | 
 | #define SMC_LLC_DEL_OP_INIT_TERM	0x00020000 | 
 | #define SMC_LLC_DEL_PROG_INIT_TERM	0x00030000 | 
 | #define SMC_LLC_DEL_PROT_VIOL		0x00040000 | 
 | #define SMC_LLC_DEL_NO_ASYM_NEEDED	0x00050000 | 
 | /* LLC DELETE LINK Response Reason Codes */ | 
 | #define SMC_LLC_DEL_NOLNK	0x00100000  /* Unknown Link ID (no link) */ | 
 | #define SMC_LLC_DEL_NOLGR	0x00200000  /* Unknown Link Group */ | 
 |  | 
 | /* returns a usable link of the link group, or NULL */ | 
 | static inline struct smc_link *smc_llc_usable_link(struct smc_link_group *lgr) | 
 | { | 
 | 	int i; | 
 |  | 
 | 	for (i = 0; i < SMC_LINKS_PER_LGR_MAX; i++) | 
 | 		if (smc_link_usable(&lgr->lnk[i])) | 
 | 			return &lgr->lnk[i]; | 
 | 	return NULL; | 
 | } | 
 |  | 
 | /* set the termination reason code for the link group */ | 
 | static inline void smc_llc_set_termination_rsn(struct smc_link_group *lgr, | 
 | 					       u32 rsn) | 
 | { | 
 | 	if (!lgr->llc_termination_rsn) | 
 | 		lgr->llc_termination_rsn = rsn; | 
 | } | 
 |  | 
 | /* transmit */ | 
 | int smc_llc_send_confirm_link(struct smc_link *lnk, | 
 | 			      enum smc_llc_reqresp reqresp); | 
 | int smc_llc_send_add_link(struct smc_link *link, u8 mac[], u8 gid[], | 
 | 			  struct smc_link *link_new, | 
 | 			  enum smc_llc_reqresp reqresp); | 
 | int smc_llc_send_delete_link(struct smc_link *link, u8 link_del_id, | 
 | 			     enum smc_llc_reqresp reqresp, bool orderly, | 
 | 			     u32 reason); | 
 | void smc_llc_srv_delete_link_local(struct smc_link *link, u8 del_link_id); | 
 | void smc_llc_lgr_init(struct smc_link_group *lgr, struct smc_sock *smc); | 
 | void smc_llc_lgr_clear(struct smc_link_group *lgr); | 
 | int smc_llc_link_init(struct smc_link *link); | 
 | void smc_llc_link_active(struct smc_link *link); | 
 | void smc_llc_link_clear(struct smc_link *link, bool log); | 
 | int smc_llc_do_confirm_rkey(struct smc_link *send_link, | 
 | 			    struct smc_buf_desc *rmb_desc); | 
 | int smc_llc_do_delete_rkey(struct smc_link_group *lgr, | 
 | 			   struct smc_buf_desc *rmb_desc); | 
 | int smc_llc_flow_initiate(struct smc_link_group *lgr, | 
 | 			  enum smc_llc_flowtype type); | 
 | void smc_llc_flow_stop(struct smc_link_group *lgr, struct smc_llc_flow *flow); | 
 | int smc_llc_eval_conf_link(struct smc_llc_qentry *qentry, | 
 | 			   enum smc_llc_reqresp type); | 
 | void smc_llc_link_set_uid(struct smc_link *link); | 
 | void smc_llc_save_peer_uid(struct smc_llc_qentry *qentry); | 
 | struct smc_llc_qentry *smc_llc_wait(struct smc_link_group *lgr, | 
 | 				    struct smc_link *lnk, | 
 | 				    int time_out, u8 exp_msg); | 
 | struct smc_llc_qentry *smc_llc_flow_qentry_clr(struct smc_llc_flow *flow); | 
 | void smc_llc_flow_qentry_del(struct smc_llc_flow *flow); | 
 | void smc_llc_send_link_delete_all(struct smc_link_group *lgr, bool ord, | 
 | 				  u32 rsn); | 
 | int smc_llc_cli_add_link(struct smc_link *link, struct smc_llc_qentry *qentry); | 
 | int smc_llc_srv_add_link(struct smc_link *link, | 
 | 			 struct smc_llc_qentry *req_qentry); | 
 | void smc_llc_add_link_local(struct smc_link *link); | 
 | int smc_llc_init(void) __init; | 
 |  | 
 | #endif /* SMC_LLC_H */ |