#ifndef _SYS_IB_MGT_IBMF_IBMF_SAA_IMPL_H
#define _SYS_IB_MGT_IBMF_IBMF_SAA_IMPL_H
#ifdef __cplusplus
extern "C" {
#endif
#include <sys/ib/mgt/ibmf/ibmf_saa.h>
#include <sys/ib/mgt/ibmf/ibmf_impl.h>
#define SAA_MAX_CLIENTS_PER_PORT 100
#define SAA_MAD_BASE_VERSION 1
#define SAA_MAD_CLASS_VERSION 2
#define IBMF_SAA_RETRANS_RETRIES 0
#define IBMF_SAA_MAX_SUBNET_TIMEOUT 20
#define IBMF_SAA_MAX_RESP_TIME 20
#define IBMF_SAA_MAX_BUSY_RETRY_COUNT 10
#define IBMF_SAA_MAX_WAIT_TIME_IN_SECS 60
#define IBMF_SAA_TRANS_WAIT_TIME_IN_SECS 240
#define IBMF_SAA_BUSY_RETRY_SLEEP_SECS 1
typedef enum saa_port_state_s {
IBMF_SAA_PORT_STATE_REGISTERING,
IBMF_SAA_PORT_STATE_READY,
IBMF_SAA_PORT_STATE_INVALID,
IBMF_SAA_PORT_STATE_PURGING
} saa_port_state_t;
typedef struct saa_port_s {
struct saa_port_s *next;
kmutex_t saa_pt_mutex;
kcondvar_t saa_pt_ibmf_reg_cv;
saa_port_state_t saa_pt_state;
int saa_pt_reference_count;
ib_guid_t saa_pt_port_guid;
ibmf_register_info_t saa_pt_ibmf_reginfo;
ibmf_handle_t saa_pt_ibmf_handle;
ibmf_impl_caps_t saa_pt_ibmf_impl_features;
ibmf_qp_handle_t saa_pt_qp_handle;
ib_qpn_t saa_pt_qpn;
int saa_pt_timeout;
uint16_t saa_pt_sa_cap_mask;
ibmf_addr_info_t saa_pt_ibmf_addr_info;
ibmf_global_addr_info_t saa_pt_ibmf_global_addr;
uint32_t saa_pt_ibmf_msg_flags;
boolean_t saa_pt_redirect_active;
ibmf_retrans_t saa_pt_ibmf_retrans;
uint64_t saa_pt_current_tid;
int saa_pt_num_outstanding_trans;
kmutex_t saa_pt_kstat_mutex;
struct kstat *saa_pt_kstatp;
kmutex_t saa_pt_event_sub_mutex;
uint8_t saa_pt_event_sub_arrive_mask;
uint8_t saa_pt_event_sub_success_mask;
uint8_t saa_pt_event_sub_last_success_mask;
struct saa_client_data_s *saa_pt_event_sub_client_list;
ib_guid_t saa_pt_node_guid;
uint8_t saa_pt_port_num;
hrtime_t saa_pt_sa_uptime;
} saa_port_t;
_NOTE(MUTEX_PROTECTS_DATA(saa_port_t::saa_pt_mutex,
saa_port_t::saa_pt_reference_count
saa_port_t::saa_pt_ibmf_reg_cv
saa_port_t::saa_pt_ibmf_retrans
saa_port_t::saa_pt_current_tid
saa_port_t::saa_pt_num_outstanding_trans
saa_port_t::saa_pt_timeout
saa_port_t::saa_pt_ibmf_addr_info
saa_port_t::saa_pt_ibmf_global_addr
saa_port_t::saa_pt_ibmf_msg_flags
saa_port_t::saa_pt_redirect_active))
_NOTE(MUTEX_PROTECTS_DATA(saa_port_t::saa_pt_kstat_mutex,
saa_port_t::saa_pt_kstatp))
#define IBMF_SAA_PORT_EVENT_SUB_ALL_ARRIVE \
(IBMF_SAA_EVENT_STATUS_MASK_PRODUCER_CA | \
IBMF_SAA_EVENT_STATUS_MASK_PRODUCER_SWITCH | \
IBMF_SAA_EVENT_STATUS_MASK_PRODUCER_ROUTER | \
IBMF_SAA_EVENT_STATUS_MASK_PRODUCER_SM)
typedef struct ibmf_saa_kstat_s {
kstat_named_t clients_registered;
kstat_named_t clients_reg_failed;
kstat_named_t outstanding_requests;
kstat_named_t total_requests;
kstat_named_t failed_requests;
kstat_named_t requests_timedout;
} ibmf_saa_kstat_t;
#define IBMF_SAA_ADD32_KSTATS(subnetp, xx, val) \
if ((subnetp != NULL) && (subnetp->saa_pt_kstatp != NULL)) { \
ibmf_saa_kstat_t *kp; \
kp = (ibmf_saa_kstat_t *)subnetp->saa_pt_kstatp->ks_data;\
kp->xx.value.ui32 += val; \
}
#define IBMF_SAA_SUB32_KSTATS(subnetp, xx, val) \
if ((subnetp != NULL) && (subnetp->saa_pt_kstatp != NULL)) { \
ibmf_saa_kstat_t *kp; \
kp = (ibmf_saa_kstat_t *)subnetp->saa_pt_kstatp->ks_data;\
kp->xx.value.ui32 -= val; \
}
typedef enum _saa_client_state_e {
SAA_CLIENT_STATE_ACTIVE,
SAA_CLIENT_STATE_WAITING,
SAA_CLIENT_STATE_CLOSED
} saa_client_state_t;
typedef struct saa_client_data_s {
void *next;
void *saa_client_sig;
saa_port_t *saa_client_port;
kmutex_t saa_client_mutex;
int saa_client_num_pending_trans;
kcondvar_t saa_client_state_cv;
saa_client_state_t saa_client_state;
ib_smkey_t saa_client_sm_key;
int saa_client_event_cb_num_active;
kcondvar_t saa_client_event_cb_cv;
ibmf_saa_subnet_event_cb_t saa_client_event_cb;
void *saa_client_event_cb_arg;
} saa_client_data_t;
_NOTE(READ_ONLY_DATA(saa_client_data_t::saa_client_port))
_NOTE(READ_ONLY_DATA(saa_client_data_t::saa_client_sig))
typedef struct saa_state_s {
saa_port_t *saa_port_list;
kmutex_t saa_port_list_mutex;
taskq_t *saa_event_taskq;
} saa_state_t;
_NOTE(MUTEX_PROTECTS_DATA(saa_state_t::saa_port_list_mutex,
saa_port_t::next
saa_state_t::saa_port_list))
_NOTE(READ_ONLY_DATA(saa_state_t::saa_event_taskq))
typedef void (*ibmf_saa_sub_cb_t) (void *, size_t, char *, int, uint32_t);
typedef struct saa_impl_trans_info_t {
saa_client_data_t *si_trans_client_data;
saa_port_t *si_trans_port;
size_t si_trans_template_length;
uint16_t si_trans_attr_id;
uint64_t si_trans_component_mask;
void *si_trans_template;
uint8_t si_trans_method;
ibmf_saa_cb_t si_trans_callback;
void *si_trans_callback_arg;
int si_trans_status;
void *si_trans_result;
size_t si_trans_length;
uint32_t si_trans_sub_producer_type;
ibmf_saa_sub_cb_t si_trans_sub_callback;
boolean_t si_trans_unseq_unsubscribe;
uint8_t si_trans_transport_flags;
uint8_t si_trans_retry_busy_count;
hrtime_t si_trans_send_time;
} saa_impl_trans_info_t;
_NOTE(SCHEME_PROTECTS_DATA("private callback arg", saa_impl_trans_info_t))
typedef struct ibmf_saa_event_taskq_args_s {
saa_client_data_t *et_client;
ibmf_saa_subnet_event_t et_subnet_event;
ibmf_saa_event_details_t *et_event_details;
ib_mad_notice_t *et_notice;
ibmf_saa_subnet_event_cb_t et_callback;
void *et_callback_arg;
} ibmf_saa_event_taskq_args_t;
_NOTE(READ_ONLY_DATA(ibmf_saa_event_taskq_args_t::et_subnet_event
ibmf_saa_event_taskq_args_t::et_event_details
ibmf_saa_event_taskq_args_t::et_client
ibmf_saa_event_taskq_args_t::et_notice
ibmf_saa_event_taskq_args_t::et_callback
ibmf_saa_event_taskq_args_t::et_callback_arg))
int ibmf_saa_impl_init();
int ibmf_saa_impl_fini();
boolean_t ibmf_saa_is_valid(saa_port_t *saa_portp, int add_ref);
void ibmf_saa_impl_purge();
int ibmf_saa_impl_add_client(saa_port_t *saa_portp);
int ibmf_saa_impl_create_port(ib_guid_t pt_guid, saa_port_t **saa_portpp);
int ibmf_saa_impl_init_kstats(saa_port_t *saa_portp);
void ibmf_saa_impl_register_failed(saa_port_t *saa_portp);
int ibmf_saa_impl_register_port(saa_port_t *saa_portp);
void ibmf_saa_impl_get_classportinfo(saa_port_t *saa_portp);
int ibmf_saa_impl_send_request(saa_impl_trans_info_t *trans_info);
void ibmf_saa_async_cb(ibmf_handle_t ibmf_handle, ibmf_msg_t *msgp,
void *args);
void
ibmf_saa_add_event_subscriber(saa_client_data_t *client,
ibmf_saa_subnet_event_args_t *event_args);
void ibmf_saa_subscribe_events(saa_port_t *saa_portp, boolean_t subscribe,
boolean_t seq_unsubscribe);
void ibmf_saa_subscribe_sm_events(saa_port_t *saa_portp);
void
ibmf_saa_notify_event_clients(saa_port_t *saa_portp,
ibmf_saa_event_details_t *event_details,
ibmf_saa_subnet_event_t subnet_event,
saa_client_data_t *registering_client);
void
ibmf_saa_report_cb(ibmf_handle_t ibmf_handle, ibmf_msg_t *msgp,
void *args);
#ifdef __cplusplus
}
#endif
#endif