efclib.h (16844B)
1/* SPDX-License-Identifier: GPL-2.0 */ 2/* 3 * Copyright (C) 2021 Broadcom. All Rights Reserved. The term 4 * “Broadcom” refers to Broadcom Inc. and/or its subsidiaries. 5 */ 6 7#ifndef __EFCLIB_H__ 8#define __EFCLIB_H__ 9 10#include "scsi/fc/fc_els.h" 11#include "scsi/fc/fc_fs.h" 12#include "scsi/fc/fc_ns.h" 13#include "scsi/fc/fc_gs.h" 14#include "scsi/fc_frame.h" 15#include "../include/efc_common.h" 16#include "../libefc_sli/sli4.h" 17 18#define EFC_SERVICE_PARMS_LENGTH 120 19#define EFC_NAME_LENGTH 32 20#define EFC_SM_NAME_LENGTH 64 21#define EFC_DISPLAY_BUS_INFO_LENGTH 16 22 23#define EFC_WWN_LENGTH 32 24 25#define EFC_FC_ELS_DEFAULT_RETRIES 3 26 27/* Timeouts */ 28#define EFC_FC_ELS_SEND_DEFAULT_TIMEOUT 0 29#define EFC_FC_FLOGI_TIMEOUT_SEC 5 30#define EFC_SHUTDOWN_TIMEOUT_USEC 30000000 31 32/* Return values for calls from base driver to libefc */ 33#define EFC_SCSI_CALL_COMPLETE 0 34#define EFC_SCSI_CALL_ASYNC 1 35 36/* Local port topology */ 37enum efc_nport_topology { 38 EFC_NPORT_TOPO_UNKNOWN = 0, 39 EFC_NPORT_TOPO_FABRIC, 40 EFC_NPORT_TOPO_P2P, 41 EFC_NPORT_TOPO_FC_AL, 42}; 43 44#define enable_target_rscn(efc) 1 45 46enum efc_node_shutd_rsn { 47 EFC_NODE_SHUTDOWN_DEFAULT = 0, 48 EFC_NODE_SHUTDOWN_EXPLICIT_LOGO, 49 EFC_NODE_SHUTDOWN_IMPLICIT_LOGO, 50}; 51 52enum efc_node_send_ls_acc { 53 EFC_NODE_SEND_LS_ACC_NONE = 0, 54 EFC_NODE_SEND_LS_ACC_PLOGI, 55 EFC_NODE_SEND_LS_ACC_PRLI, 56}; 57 58#define EFC_LINK_STATUS_UP 0 59#define EFC_LINK_STATUS_DOWN 1 60 61/* State machine context header */ 62struct efc_sm_ctx { 63 void (*current_state)(struct efc_sm_ctx *ctx, 64 u32 evt, void *arg); 65 66 const char *description; 67 void *app; 68}; 69 70/* Description of discovered Fabric Domain */ 71struct efc_domain_record { 72 u32 index; 73 u32 priority; 74 u8 address[6]; 75 u8 wwn[8]; 76 union { 77 u8 vlan[512]; 78 u8 loop[128]; 79 } map; 80 u32 speed; 81 u32 fc_id; 82 bool is_loop; 83 bool is_nport; 84}; 85 86/* Domain events */ 87enum efc_hw_domain_event { 88 EFC_HW_DOMAIN_ALLOC_OK, 89 EFC_HW_DOMAIN_ALLOC_FAIL, 90 EFC_HW_DOMAIN_ATTACH_OK, 91 EFC_HW_DOMAIN_ATTACH_FAIL, 92 EFC_HW_DOMAIN_FREE_OK, 93 EFC_HW_DOMAIN_FREE_FAIL, 94 EFC_HW_DOMAIN_LOST, 95 EFC_HW_DOMAIN_FOUND, 96 EFC_HW_DOMAIN_CHANGED, 97}; 98 99/** 100 * Fibre Channel port object 101 * 102 * @list_entry: nport list entry 103 * @ref: reference count, each node takes a reference 104 * @release: function to free nport object 105 * @efc: pointer back to efc 106 * @instance_index: unique instance index value 107 * @display_name: port display name 108 * @is_vport: Is NPIV port 109 * @free_req_pending: pending request to free resources 110 * @attached: mark attached if reg VPI succeeds 111 * @p2p_winner: TRUE if we're the point-to-point winner 112 * @domain: pointer back to domain 113 * @wwpn: port wwpn 114 * @wwnn: port wwnn 115 * @tgt_data: target backend private port data 116 * @ini_data: initiator backend private port data 117 * @indicator: VPI 118 * @fc_id: port FC address 119 * @dma: memory for Service Parameters 120 * @wwnn_str: wwpn string 121 * @sli_wwpn: SLI provided wwpn 122 * @sli_wwnn: SLI provided wwnn 123 * @sm: nport state machine context 124 * @lookup: fc_id to node lookup object 125 * @enable_ini: SCSI initiator enabled for this port 126 * @enable_tgt: SCSI target enabled for this port 127 * @enable_rscn: port will be expecting RSCN 128 * @shutting_down: nport in process of shutting down 129 * @p2p_port_id: our port id for point-to-point 130 * @topology: topology: fabric/p2p/unknown 131 * @service_params: login parameters 132 * @p2p_remote_port_id: remote node's port id for point-to-point 133 */ 134 135struct efc_nport { 136 struct list_head list_entry; 137 struct kref ref; 138 void (*release)(struct kref *arg); 139 struct efc *efc; 140 u32 instance_index; 141 char display_name[EFC_NAME_LENGTH]; 142 bool is_vport; 143 bool free_req_pending; 144 bool attached; 145 bool attaching; 146 bool p2p_winner; 147 struct efc_domain *domain; 148 u64 wwpn; 149 u64 wwnn; 150 void *tgt_data; 151 void *ini_data; 152 153 u32 indicator; 154 u32 fc_id; 155 struct efc_dma dma; 156 157 u8 wwnn_str[EFC_WWN_LENGTH]; 158 __be64 sli_wwpn; 159 __be64 sli_wwnn; 160 161 struct efc_sm_ctx sm; 162 struct xarray lookup; 163 bool enable_ini; 164 bool enable_tgt; 165 bool enable_rscn; 166 bool shutting_down; 167 u32 p2p_port_id; 168 enum efc_nport_topology topology; 169 u8 service_params[EFC_SERVICE_PARMS_LENGTH]; 170 u32 p2p_remote_port_id; 171}; 172 173/** 174 * Fibre Channel domain object 175 * 176 * This object is a container for the various SLI components needed 177 * to connect to the domain of a FC or FCoE switch 178 * @efc: pointer back to efc 179 * @instance_index: unique instance index value 180 * @display_name: Node display name 181 * @nport_list: linked list of nports associated with this domain 182 * @ref: Reference count, each nport takes a reference 183 * @release: Function to free domain object 184 * @ini_domain: initiator backend private domain data 185 * @tgt_domain: target backend private domain data 186 * @sm: state machine context 187 * @fcf: FC Forwarder table index 188 * @fcf_indicator: FCFI 189 * @indicator: VFI 190 * @nport_count: Number of nports allocated 191 * @dma: memory for Service Parameters 192 * @fcf_wwn: WWN for FCF/switch 193 * @drvsm: driver domain sm context 194 * @attached: set true after attach completes 195 * @is_fc: is FC 196 * @is_loop: is loop topology 197 * @is_nlport: is public loop 198 * @domain_found_pending:A domain found is pending, drec is updated 199 * @req_domain_free: True if domain object should be free'd 200 * @req_accept_frames: set in domain state machine to enable frames 201 * @domain_notify_pend: Set in domain SM to avoid duplicate node event post 202 * @pending_drec: Pending drec if a domain found is pending 203 * @service_params: any nports service parameters 204 * @flogi_service_params:Fabric/P2p service parameters from FLOGI 205 * @lookup: d_id to node lookup object 206 * @nport: Pointer to first (physical) SLI port 207 */ 208struct efc_domain { 209 struct efc *efc; 210 char display_name[EFC_NAME_LENGTH]; 211 struct list_head nport_list; 212 struct kref ref; 213 void (*release)(struct kref *arg); 214 void *ini_domain; 215 void *tgt_domain; 216 217 /* Declarations private to HW/SLI */ 218 u32 fcf; 219 u32 fcf_indicator; 220 u32 indicator; 221 u32 nport_count; 222 struct efc_dma dma; 223 224 /* Declarations private to FC trannport */ 225 u64 fcf_wwn; 226 struct efc_sm_ctx drvsm; 227 bool attached; 228 bool is_fc; 229 bool is_loop; 230 bool is_nlport; 231 bool domain_found_pending; 232 bool req_domain_free; 233 bool req_accept_frames; 234 bool domain_notify_pend; 235 236 struct efc_domain_record pending_drec; 237 u8 service_params[EFC_SERVICE_PARMS_LENGTH]; 238 u8 flogi_service_params[EFC_SERVICE_PARMS_LENGTH]; 239 240 struct xarray lookup; 241 242 struct efc_nport *nport; 243}; 244 245/** 246 * Remote Node object 247 * 248 * This object represents a connection between the SLI port and another 249 * Nx_Port on the fabric. Note this can be either a well known port such 250 * as a F_Port (i.e. ff:ff:fe) or another N_Port. 251 * @indicator: RPI 252 * @fc_id: FC address 253 * @attached: true if attached 254 * @nport: associated SLI port 255 * @node: associated node 256 */ 257struct efc_remote_node { 258 u32 indicator; 259 u32 index; 260 u32 fc_id; 261 262 bool attached; 263 264 struct efc_nport *nport; 265 void *node; 266}; 267 268/** 269 * FC Node object 270 * @efc: pointer back to efc structure 271 * @display_name: Node display name 272 * @nort: Assosiated nport pointer. 273 * @hold_frames: hold incoming frames if true 274 * @els_io_enabled: Enable allocating els ios for this node 275 * @els_ios_lock: lock to protect the els ios list 276 * @els_ios_list: ELS I/O's for this node 277 * @ini_node: backend initiator private node data 278 * @tgt_node: backend target private node data 279 * @rnode: Remote node 280 * @sm: state machine context 281 * @evtdepth: current event posting nesting depth 282 * @req_free: this node is to be free'd 283 * @attached: node is attached (REGLOGIN complete) 284 * @fcp_enabled: node is enabled to handle FCP 285 * @rscn_pending: for name server node RSCN is pending 286 * @send_plogi: send PLOGI accept, upon completion of node attach 287 * @send_plogi_acc: TRUE if io_alloc() is enabled. 288 * @send_ls_acc: type of LS acc to send 289 * @ls_acc_io: SCSI IO for LS acc 290 * @ls_acc_oxid: OX_ID for pending accept 291 * @ls_acc_did: D_ID for pending accept 292 * @shutdown_reason: reason for node shutdown 293 * @sparm_dma_buf: service parameters buffer 294 * @service_params: plogi/acc frame from remote device 295 * @pend_frames_lock: lock for inbound pending frames list 296 * @pend_frames: inbound pending frames list 297 * @pend_frames_processed:count of frames processed in hold frames interval 298 * @ox_id_in_use: used to verify one at a time us of ox_id 299 * @els_retries_remaining:for ELS, number of retries remaining 300 * @els_req_cnt: number of outstanding ELS requests 301 * @els_cmpl_cnt: number of outstanding ELS completions 302 * @abort_cnt: Abort counter for debugging purpos 303 * @current_state_name: current node state 304 * @prev_state_name: previous node state 305 * @current_evt: current event 306 * @prev_evt: previous event 307 * @targ: node is target capable 308 * @init: node is init capable 309 * @refound: Handle node refound case when node is being deleted 310 * @els_io_pend_list: list of pending (not yet processed) ELS IOs 311 * @els_io_active_list: list of active (processed) ELS IOs 312 * @nodedb_state: Node debugging, saved state 313 * @gidpt_delay_timer: GIDPT delay timer 314 * @time_last_gidpt_msec:Start time of last target RSCN GIDPT 315 * @wwnn: remote port WWNN 316 * @wwpn: remote port WWPN 317 */ 318struct efc_node { 319 struct efc *efc; 320 char display_name[EFC_NAME_LENGTH]; 321 struct efc_nport *nport; 322 struct kref ref; 323 void (*release)(struct kref *arg); 324 bool hold_frames; 325 bool els_io_enabled; 326 bool send_plogi_acc; 327 bool send_plogi; 328 bool rscn_pending; 329 bool fcp_enabled; 330 bool attached; 331 bool req_free; 332 333 spinlock_t els_ios_lock; 334 struct list_head els_ios_list; 335 void *ini_node; 336 void *tgt_node; 337 338 struct efc_remote_node rnode; 339 /* Declarations private to FC trannport */ 340 struct efc_sm_ctx sm; 341 u32 evtdepth; 342 343 enum efc_node_send_ls_acc send_ls_acc; 344 void *ls_acc_io; 345 u32 ls_acc_oxid; 346 u32 ls_acc_did; 347 enum efc_node_shutd_rsn shutdown_reason; 348 bool targ; 349 bool init; 350 bool refound; 351 struct efc_dma sparm_dma_buf; 352 u8 service_params[EFC_SERVICE_PARMS_LENGTH]; 353 spinlock_t pend_frames_lock; 354 struct list_head pend_frames; 355 u32 pend_frames_processed; 356 u32 ox_id_in_use; 357 u32 els_retries_remaining; 358 u32 els_req_cnt; 359 u32 els_cmpl_cnt; 360 u32 abort_cnt; 361 362 char current_state_name[EFC_SM_NAME_LENGTH]; 363 char prev_state_name[EFC_SM_NAME_LENGTH]; 364 int current_evt; 365 int prev_evt; 366 367 void (*nodedb_state)(struct efc_sm_ctx *ctx, 368 u32 evt, void *arg); 369 struct timer_list gidpt_delay_timer; 370 u64 time_last_gidpt_msec; 371 372 char wwnn[EFC_WWN_LENGTH]; 373 char wwpn[EFC_WWN_LENGTH]; 374}; 375 376/** 377 * NPIV port 378 * 379 * Collection of the information required to restore a virtual port across 380 * link events 381 * @wwnn: node name 382 * @wwpn: port name 383 * @fc_id: port id 384 * @tgt_data: target backend pointer 385 * @ini_data: initiator backend pointe 386 * @nport: Used to match record after attaching for update 387 * 388 */ 389 390struct efc_vport { 391 struct list_head list_entry; 392 u64 wwnn; 393 u64 wwpn; 394 u32 fc_id; 395 bool enable_tgt; 396 bool enable_ini; 397 void *tgt_data; 398 void *ini_data; 399 struct efc_nport *nport; 400}; 401 402#define node_printf(node, fmt, args...) \ 403 efc_log_info(node->efc, "[%s] " fmt, node->display_name, ##args) 404 405/* Node SM IO Context Callback structure */ 406struct efc_node_cb { 407 int status; 408 int ext_status; 409 struct efc_hw_rq_buffer *header; 410 struct efc_hw_rq_buffer *payload; 411 struct efc_dma els_rsp; 412 413 /* Actual length of data received */ 414 int rsp_len; 415}; 416 417struct efc_hw_rq_buffer { 418 u16 rqindex; 419 struct efc_dma dma; 420}; 421 422/** 423 * FC sequence object 424 * 425 * Defines a general FC sequence object 426 * @hw: HW that owns this sequence 427 * @fcfi: FCFI associated with sequence 428 * @header: Received frame header 429 * @payload: Received frame header 430 * @hw_priv: HW private context 431 */ 432struct efc_hw_sequence { 433 struct list_head list_entry; 434 void *hw; 435 u8 fcfi; 436 struct efc_hw_rq_buffer *header; 437 struct efc_hw_rq_buffer *payload; 438 void *hw_priv; 439}; 440 441enum efc_disc_io_type { 442 EFC_DISC_IO_ELS_REQ, 443 EFC_DISC_IO_ELS_RESP, 444 EFC_DISC_IO_CT_REQ, 445 EFC_DISC_IO_CT_RESP 446}; 447 448struct efc_io_els_params { 449 u32 s_id; 450 u16 ox_id; 451 u8 timeout; 452}; 453 454struct efc_io_ct_params { 455 u8 r_ctl; 456 u8 type; 457 u8 df_ctl; 458 u8 timeout; 459 u16 ox_id; 460}; 461 462union efc_disc_io_param { 463 struct efc_io_els_params els; 464 struct efc_io_ct_params ct; 465}; 466 467struct efc_disc_io { 468 struct efc_dma req; /* send buffer */ 469 struct efc_dma rsp; /* receive buffer */ 470 enum efc_disc_io_type io_type; /* EFC_DISC_IO_TYPE enum*/ 471 u16 xmit_len; /* Length of els request*/ 472 u16 rsp_len; /* Max length of rsps to be rcvd */ 473 u32 rpi; /* Registered RPI */ 474 u32 vpi; /* VPI for this nport */ 475 u32 s_id; 476 u32 d_id; 477 bool rpi_registered; /* if false, use tmp RPI */ 478 union efc_disc_io_param iparam; 479}; 480 481/* Return value indiacating the sequence can not be freed */ 482#define EFC_HW_SEQ_HOLD 0 483/* Return value indiacating the sequence can be freed */ 484#define EFC_HW_SEQ_FREE 1 485 486struct libefc_function_template { 487 /*Sport*/ 488 int (*new_nport)(struct efc *efc, struct efc_nport *sp); 489 void (*del_nport)(struct efc *efc, struct efc_nport *sp); 490 491 /*Scsi Node*/ 492 int (*scsi_new_node)(struct efc *efc, struct efc_node *n); 493 int (*scsi_del_node)(struct efc *efc, struct efc_node *n, int reason); 494 495 int (*issue_mbox_rqst)(void *efct, void *buf, void *cb, void *arg); 496 /*Send ELS IO*/ 497 int (*send_els)(struct efc *efc, struct efc_disc_io *io); 498 /*Send BLS IO*/ 499 int (*send_bls)(struct efc *efc, u32 type, struct sli_bls_params *bls); 500 /*Free HW frame*/ 501 int (*hw_seq_free)(struct efc *efc, struct efc_hw_sequence *seq); 502}; 503 504#define EFC_LOG_LIB 0x01 505#define EFC_LOG_NODE 0x02 506#define EFC_LOG_PORT 0x04 507#define EFC_LOG_DOMAIN 0x08 508#define EFC_LOG_ELS 0x10 509#define EFC_LOG_DOMAIN_SM 0x20 510#define EFC_LOG_SM 0x40 511 512/* efc library port structure */ 513struct efc { 514 void *base; 515 struct pci_dev *pci; 516 struct sli4 *sli; 517 u32 fcfi; 518 u64 req_wwpn; 519 u64 req_wwnn; 520 521 u64 def_wwpn; 522 u64 def_wwnn; 523 u64 max_xfer_size; 524 mempool_t *node_pool; 525 struct dma_pool *node_dma_pool; 526 u32 nodes_count; 527 528 u32 link_status; 529 530 struct list_head vport_list; 531 /* lock to protect the vport list */ 532 spinlock_t vport_lock; 533 534 struct libefc_function_template tt; 535 /* lock to protect the discovery library. 536 * Refer to efclib.c for more details. 537 */ 538 spinlock_t lock; 539 540 bool enable_ini; 541 bool enable_tgt; 542 543 u32 log_level; 544 545 struct efc_domain *domain; 546 void (*domain_free_cb)(struct efc *efc, void *arg); 547 void *domain_free_cb_arg; 548 549 u64 tgt_rscn_delay_msec; 550 u64 tgt_rscn_period_msec; 551 552 bool external_loopback; 553 u32 nodedb_mask; 554 u32 logmask; 555 mempool_t *els_io_pool; 556 atomic_t els_io_alloc_failed_count; 557 558 /* hold pending frames */ 559 bool hold_frames; 560 /* lock to protect pending frames list access */ 561 spinlock_t pend_frames_lock; 562 struct list_head pend_frames; 563 /* count of pending frames that were processed */ 564 u32 pend_frames_processed; 565 566}; 567 568/* 569 * EFC library registration 570 * **********************************/ 571int efcport_init(struct efc *efc); 572void efcport_destroy(struct efc *efc); 573/* 574 * EFC Domain 575 * **********************************/ 576int efc_domain_cb(void *arg, int event, void *data); 577void 578efc_register_domain_free_cb(struct efc *efc, 579 void (*callback)(struct efc *efc, void *arg), 580 void *arg); 581 582/* 583 * EFC nport 584 * **********************************/ 585void efc_nport_cb(void *arg, int event, void *data); 586struct efc_vport * 587efc_vport_create_spec(struct efc *efc, u64 wwnn, u64 wwpn, u32 fc_id, 588 bool enable_ini, bool enable_tgt, 589 void *tgt_data, void *ini_data); 590int efc_nport_vport_new(struct efc_domain *domain, u64 wwpn, 591 u64 wwnn, u32 fc_id, bool ini, bool tgt, 592 void *tgt_data, void *ini_data); 593int efc_nport_vport_del(struct efc *efc, struct efc_domain *domain, 594 u64 wwpn, u64 wwnn); 595 596void efc_vport_del_all(struct efc *efc); 597 598/* 599 * EFC Node 600 * **********************************/ 601int efc_remote_node_cb(void *arg, int event, void *data); 602void efc_node_fcid_display(u32 fc_id, char *buffer, u32 buf_len); 603void efc_node_post_shutdown(struct efc_node *node, void *arg); 604u64 efc_node_get_wwpn(struct efc_node *node); 605 606/* 607 * EFC FCP/ELS/CT interface 608 * **********************************/ 609void efc_dispatch_frame(struct efc *efc, struct efc_hw_sequence *seq); 610void efc_disc_io_complete(struct efc_disc_io *io, u32 len, u32 status, 611 u32 ext_status); 612 613/* 614 * EFC SCSI INTERACTION LAYER 615 * **********************************/ 616void efc_scsi_sess_reg_complete(struct efc_node *node, u32 status); 617void efc_scsi_del_initiator_complete(struct efc *efc, struct efc_node *node); 618void efc_scsi_del_target_complete(struct efc *efc, struct efc_node *node); 619void efc_scsi_io_list_empty(struct efc *efc, struct efc_node *node); 620 621#endif /* __EFCLIB_H__ */