1 /*- 2 * Copyright (c) 2017 Broadcom. All rights reserved. 3 * The term "Broadcom" refers to Broadcom Limited and/or its subsidiaries. 4 * 5 * Redistribution and use in source and binary forms, with or without 6 * modification, are permitted provided that the following conditions are met: 7 * 8 * 1. Redistributions of source code must retain the above copyright notice, 9 * this list of conditions and the following disclaimer. 10 * 11 * 2. Redistributions in binary form must reproduce the above copyright notice, 12 * this list of conditions and the following disclaimer in the documentation 13 * and/or other materials provided with the distribution. 14 * 15 * 3. Neither the name of the copyright holder nor the names of its contributors 16 * may be used to endorse or promote products derived from this software 17 * without specific prior written permission. 18 * 19 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" 20 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 21 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 22 * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE 23 * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR 24 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF 25 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS 26 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN 27 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) 28 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE 29 * POSSIBILITY OF SUCH DAMAGE. 30 * 31 * $FreeBSD$ 32 */ 33 34 /** 35 * @file 36 * Defines and implements the Hardware Abstraction Layer (HW). 37 * All interaction with the hardware is performed through the HW, which abstracts 38 * the details of the underlying SLI-4 implementation. 39 */ 40 41 /** 42 * @defgroup devInitShutdown Device Initialization and Shutdown 43 * @defgroup domain Domain Functions 44 * @defgroup port Port Functions 45 * @defgroup node Remote Node Functions 46 * @defgroup io IO Functions 47 * @defgroup interrupt Interrupt handling 48 * @defgroup os OS Required Functions 49 */ 50 51 #include "ocs.h" 52 #include "ocs_os.h" 53 #include "ocs_hw.h" 54 #include "ocs_hw_queues.h" 55 56 #define OCS_HW_MQ_DEPTH 128 57 #define OCS_HW_READ_FCF_SIZE 4096 58 #define OCS_HW_DEFAULT_AUTO_XFER_RDY_IOS 256 59 #define OCS_HW_WQ_TIMER_PERIOD_MS 500 60 61 /* values used for setting the auto xfer rdy parameters */ 62 #define OCS_HW_AUTO_XFER_RDY_BLK_SIZE_DEFAULT 0 /* 512 bytes */ 63 #define OCS_HW_AUTO_XFER_RDY_REF_TAG_IS_LBA_DEFAULT TRUE 64 #define OCS_HW_AUTO_XFER_RDY_APP_TAG_VALID_DEFAULT FALSE 65 #define OCS_HW_AUTO_XFER_RDY_APP_TAG_VALUE_DEFAULT 0 66 #define OCS_HW_REQUE_XRI_REGTAG 65534 67 /* max command and response buffer lengths -- arbitrary at the moment */ 68 #define OCS_HW_DMTF_CLP_CMD_MAX 256 69 #define OCS_HW_DMTF_CLP_RSP_MAX 256 70 71 /* HW global data */ 72 ocs_hw_global_t hw_global; 73 74 static void ocs_hw_queue_hash_add(ocs_queue_hash_t *, uint16_t, uint16_t); 75 static void ocs_hw_adjust_wqs(ocs_hw_t *hw); 76 static uint32_t ocs_hw_get_num_chutes(ocs_hw_t *hw); 77 static int32_t ocs_hw_cb_link(void *, void *); 78 static int32_t ocs_hw_cb_fip(void *, void *); 79 static int32_t ocs_hw_command_process(ocs_hw_t *, int32_t, uint8_t *, size_t); 80 static int32_t ocs_hw_mq_process(ocs_hw_t *, int32_t, sli4_queue_t *); 81 static int32_t ocs_hw_cb_read_fcf(ocs_hw_t *, int32_t, uint8_t *, void *); 82 static int32_t ocs_hw_cb_node_attach(ocs_hw_t *, int32_t, uint8_t *, void *); 83 static int32_t ocs_hw_cb_node_free(ocs_hw_t *, int32_t, uint8_t *, void *); 84 static int32_t ocs_hw_cb_node_free_all(ocs_hw_t *, int32_t, uint8_t *, void *); 85 static ocs_hw_rtn_e ocs_hw_setup_io(ocs_hw_t *); 86 static ocs_hw_rtn_e ocs_hw_init_io(ocs_hw_t *); 87 static int32_t ocs_hw_flush(ocs_hw_t *); 88 static int32_t ocs_hw_command_cancel(ocs_hw_t *); 89 static int32_t ocs_hw_io_cancel(ocs_hw_t *); 90 static void ocs_hw_io_quarantine(ocs_hw_t *hw, hw_wq_t *wq, ocs_hw_io_t *io); 91 static void ocs_hw_io_restore_sgl(ocs_hw_t *, ocs_hw_io_t *); 92 static int32_t ocs_hw_io_ini_sge(ocs_hw_t *, ocs_hw_io_t *, ocs_dma_t *, uint32_t, ocs_dma_t *); 93 static ocs_hw_rtn_e ocs_hw_firmware_write_lancer(ocs_hw_t *hw, ocs_dma_t *dma, uint32_t size, uint32_t offset, int last, ocs_hw_fw_cb_t cb, void *arg); 94 static int32_t ocs_hw_cb_fw_write(ocs_hw_t *, int32_t, uint8_t *, void *); 95 static int32_t ocs_hw_cb_sfp(ocs_hw_t *, int32_t, uint8_t *, void *); 96 static int32_t ocs_hw_cb_temp(ocs_hw_t *, int32_t, uint8_t *, void *); 97 static int32_t ocs_hw_cb_link_stat(ocs_hw_t *, int32_t, uint8_t *, void *); 98 static int32_t ocs_hw_cb_host_stat(ocs_hw_t *hw, int32_t status, uint8_t *mqe, void *arg); 99 static void ocs_hw_dmtf_clp_cb(ocs_hw_t *hw, int32_t status, uint8_t *mqe, void *arg); 100 static int32_t ocs_hw_clp_resp_get_value(ocs_hw_t *hw, const char *keyword, char *value, uint32_t value_len, const char *resp, uint32_t resp_len); 101 typedef void (*ocs_hw_dmtf_clp_cb_t)(ocs_hw_t *hw, int32_t status, uint32_t result_len, void *arg); 102 static ocs_hw_rtn_e ocs_hw_exec_dmtf_clp_cmd(ocs_hw_t *hw, ocs_dma_t *dma_cmd, ocs_dma_t *dma_resp, uint32_t opts, ocs_hw_dmtf_clp_cb_t cb, void *arg); 103 static void ocs_hw_linkcfg_dmtf_clp_cb(ocs_hw_t *hw, int32_t status, uint32_t result_len, void *arg); 104 105 static int32_t __ocs_read_topology_cb(ocs_hw_t *, int32_t, uint8_t *, void *); 106 static ocs_hw_rtn_e ocs_hw_get_linkcfg(ocs_hw_t *, uint32_t, ocs_hw_port_control_cb_t, void *); 107 static ocs_hw_rtn_e ocs_hw_get_linkcfg_lancer(ocs_hw_t *, uint32_t, ocs_hw_port_control_cb_t, void *); 108 static ocs_hw_rtn_e ocs_hw_get_linkcfg_skyhawk(ocs_hw_t *, uint32_t, ocs_hw_port_control_cb_t, void *); 109 static ocs_hw_rtn_e ocs_hw_set_linkcfg(ocs_hw_t *, ocs_hw_linkcfg_e, uint32_t, ocs_hw_port_control_cb_t, void *); 110 static ocs_hw_rtn_e ocs_hw_set_linkcfg_lancer(ocs_hw_t *, ocs_hw_linkcfg_e, uint32_t, ocs_hw_port_control_cb_t, void *); 111 static ocs_hw_rtn_e ocs_hw_set_linkcfg_skyhawk(ocs_hw_t *, ocs_hw_linkcfg_e, uint32_t, ocs_hw_port_control_cb_t, void *); 112 static void ocs_hw_init_linkcfg_cb(int32_t status, uintptr_t value, void *arg); 113 static ocs_hw_rtn_e ocs_hw_set_eth_license(ocs_hw_t *hw, uint32_t license); 114 static ocs_hw_rtn_e ocs_hw_set_dif_seed(ocs_hw_t *hw); 115 static ocs_hw_rtn_e ocs_hw_set_dif_mode(ocs_hw_t *hw); 116 static void ocs_hw_io_free_internal(void *arg); 117 static void ocs_hw_io_free_port_owned(void *arg); 118 static ocs_hw_rtn_e ocs_hw_config_auto_xfer_rdy_t10pi(ocs_hw_t *hw, uint8_t *buf); 119 static ocs_hw_rtn_e ocs_hw_config_set_fdt_xfer_hint(ocs_hw_t *hw, uint32_t fdt_xfer_hint); 120 static void ocs_hw_wq_process_abort(void *arg, uint8_t *cqe, int32_t status); 121 static int32_t ocs_hw_config_mrq(ocs_hw_t *hw, uint8_t, uint16_t, uint16_t); 122 static ocs_hw_rtn_e ocs_hw_config_watchdog_timer(ocs_hw_t *hw); 123 static ocs_hw_rtn_e ocs_hw_config_sli_port_health_check(ocs_hw_t *hw, uint8_t query, uint8_t enable); 124 125 /* HW domain database operations */ 126 static int32_t ocs_hw_domain_add(ocs_hw_t *, ocs_domain_t *); 127 static int32_t ocs_hw_domain_del(ocs_hw_t *, ocs_domain_t *); 128 129 /* Port state machine */ 130 static void *__ocs_hw_port_alloc_init(ocs_sm_ctx_t *, ocs_sm_event_t, void *); 131 static void *__ocs_hw_port_alloc_read_sparm64(ocs_sm_ctx_t *, ocs_sm_event_t, void *); 132 static void *__ocs_hw_port_alloc_init_vpi(ocs_sm_ctx_t *, ocs_sm_event_t, void *); 133 static void *__ocs_hw_port_done(ocs_sm_ctx_t *, ocs_sm_event_t, void *); 134 static void *__ocs_hw_port_free_unreg_vpi(ocs_sm_ctx_t *, ocs_sm_event_t, void *); 135 136 /* Domain state machine */ 137 static void *__ocs_hw_domain_init(ocs_sm_ctx_t *, ocs_sm_event_t, void *); 138 static void *__ocs_hw_domain_alloc_reg_fcfi(ocs_sm_ctx_t *, ocs_sm_event_t, void *); 139 static void * __ocs_hw_domain_alloc_init_vfi(ocs_sm_ctx_t *, ocs_sm_event_t, void *); 140 static void *__ocs_hw_domain_free_unreg_vfi(ocs_sm_ctx_t *, ocs_sm_event_t, void *); 141 static void *__ocs_hw_domain_free_unreg_fcfi(ocs_sm_ctx_t *ctx, ocs_sm_event_t evt, void *data); 142 static int32_t __ocs_hw_domain_cb(ocs_hw_t *, int32_t, uint8_t *, void *); 143 static int32_t __ocs_hw_port_cb(ocs_hw_t *, int32_t, uint8_t *, void *); 144 static int32_t __ocs_hw_port_realloc_cb(ocs_hw_t *hw, int32_t status, uint8_t *mqe, void *arg); 145 146 /* BZ 161832 */ 147 static void ocs_hw_check_sec_hio_list(ocs_hw_t *hw); 148 149 /* WQE timeouts */ 150 static void target_wqe_timer_cb(void *arg); 151 static void shutdown_target_wqe_timer(ocs_hw_t *hw); 152 153 static inline void 154 ocs_hw_add_io_timed_wqe(ocs_hw_t *hw, ocs_hw_io_t *io) 155 { 156 if (hw->config.emulate_tgt_wqe_timeout && io->tgt_wqe_timeout) { 157 /* 158 * Active WQE list currently only used for 159 * target WQE timeouts. 160 */ 161 ocs_lock(&hw->io_lock); 162 ocs_list_add_tail(&hw->io_timed_wqe, io); 163 io->submit_ticks = ocs_get_os_ticks(); 164 ocs_unlock(&hw->io_lock); 165 } 166 } 167 168 static inline void 169 ocs_hw_remove_io_timed_wqe(ocs_hw_t *hw, ocs_hw_io_t *io) 170 { 171 if (hw->config.emulate_tgt_wqe_timeout) { 172 /* 173 * If target wqe timeouts are enabled, 174 * remove from active wqe list. 175 */ 176 ocs_lock(&hw->io_lock); 177 if (ocs_list_on_list(&io->wqe_link)) { 178 ocs_list_remove(&hw->io_timed_wqe, io); 179 } 180 ocs_unlock(&hw->io_lock); 181 } 182 } 183 184 static uint8_t ocs_hw_iotype_is_originator(uint16_t io_type) 185 { 186 switch (io_type) { 187 case OCS_HW_IO_INITIATOR_READ: 188 case OCS_HW_IO_INITIATOR_WRITE: 189 case OCS_HW_IO_INITIATOR_NODATA: 190 case OCS_HW_FC_CT: 191 case OCS_HW_ELS_REQ: 192 return 1; 193 default: 194 return 0; 195 } 196 } 197 198 static uint8_t ocs_hw_wcqe_abort_needed(uint16_t status, uint8_t ext, uint8_t xb) 199 { 200 /* if exchange not active, nothing to abort */ 201 if (!xb) { 202 return FALSE; 203 } 204 if (status == SLI4_FC_WCQE_STATUS_LOCAL_REJECT) { 205 switch (ext) { 206 /* exceptions where abort is not needed */ 207 case SLI4_FC_LOCAL_REJECT_INVALID_RPI: /* lancer returns this after unreg_rpi */ 208 case SLI4_FC_LOCAL_REJECT_ABORT_REQUESTED: /* abort already in progress */ 209 return FALSE; 210 default: 211 break; 212 } 213 } 214 return TRUE; 215 } 216 217 /** 218 * @brief Determine the number of chutes on the device. 219 * 220 * @par Description 221 * Some devices require queue resources allocated per protocol processor 222 * (chute). This function returns the number of chutes on this device. 223 * 224 * @param hw Hardware context allocated by the caller. 225 * 226 * @return Returns the number of chutes on the device for protocol. 227 */ 228 static uint32_t 229 ocs_hw_get_num_chutes(ocs_hw_t *hw) 230 { 231 uint32_t num_chutes = 1; 232 233 if (sli_get_is_dual_ulp_capable(&hw->sli) && 234 sli_get_is_ulp_enabled(&hw->sli, 0) && 235 sli_get_is_ulp_enabled(&hw->sli, 1)) { 236 num_chutes = 2; 237 } 238 return num_chutes; 239 } 240 241 static ocs_hw_rtn_e 242 ocs_hw_link_event_init(ocs_hw_t *hw) 243 { 244 ocs_hw_assert(hw); 245 246 hw->link.status = SLI_LINK_STATUS_MAX; 247 hw->link.topology = SLI_LINK_TOPO_NONE; 248 hw->link.medium = SLI_LINK_MEDIUM_MAX; 249 hw->link.speed = 0; 250 hw->link.loop_map = NULL; 251 hw->link.fc_id = UINT32_MAX; 252 253 return OCS_HW_RTN_SUCCESS; 254 } 255 256 /** 257 * @ingroup devInitShutdown 258 * @brief If this is physical port 0, then read the max dump size. 259 * 260 * @par Description 261 * Queries the FW for the maximum dump size 262 * 263 * @param hw Hardware context allocated by the caller. 264 * 265 * @return Returns 0 on success, or a non-zero value on failure. 266 */ 267 static ocs_hw_rtn_e 268 ocs_hw_read_max_dump_size(ocs_hw_t *hw) 269 { 270 uint8_t buf[SLI4_BMBX_SIZE]; 271 uint8_t bus, dev, func; 272 int rc; 273 274 /* lancer only */ 275 if ((SLI4_IF_TYPE_LANCER_FC_ETH != sli_get_if_type(&hw->sli)) && 276 (SLI4_IF_TYPE_LANCER_G7 != sli_get_if_type(&hw->sli))) { 277 ocs_log_debug(hw->os, "Function only supported for I/F type 2\n"); 278 return OCS_HW_RTN_ERROR; 279 } 280 281 /* 282 * Make sure the FW is new enough to support this command. If the FW 283 * is too old, the FW will UE. 284 */ 285 if (hw->workaround.disable_dump_loc) { 286 ocs_log_test(hw->os, "FW version is too old for this feature\n"); 287 return OCS_HW_RTN_ERROR; 288 } 289 290 /* attempt to detemine the dump size for function 0 only. */ 291 ocs_get_bus_dev_func(hw->os, &bus, &dev, &func); 292 if (func == 0) { 293 if (sli_cmd_common_set_dump_location(&hw->sli, buf, 294 SLI4_BMBX_SIZE, 1, 0, NULL, 0)) { 295 sli4_res_common_set_dump_location_t *rsp = 296 (sli4_res_common_set_dump_location_t *) 297 (buf + offsetof(sli4_cmd_sli_config_t, 298 payload.embed)); 299 300 rc = ocs_hw_command(hw, buf, OCS_CMD_POLL, NULL, NULL); 301 if (rc != OCS_HW_RTN_SUCCESS) { 302 ocs_log_test(hw->os, "set dump location command failed\n"); 303 return rc; 304 } else { 305 hw->dump_size = rsp->buffer_length; 306 ocs_log_debug(hw->os, "Dump size %x\n", rsp->buffer_length); 307 } 308 } 309 } 310 return OCS_HW_RTN_SUCCESS; 311 } 312 313 /** 314 * @ingroup devInitShutdown 315 * @brief Set up the Hardware Abstraction Layer module. 316 * 317 * @par Description 318 * Calls set up to configure the hardware. 319 * 320 * @param hw Hardware context allocated by the caller. 321 * @param os Device abstraction. 322 * @param port_type Protocol type of port, such as FC and NIC. 323 * 324 * @todo Why is port_type a parameter? 325 * 326 * @return Returns 0 on success, or a non-zero value on failure. 327 */ 328 ocs_hw_rtn_e 329 ocs_hw_setup(ocs_hw_t *hw, ocs_os_handle_t os, sli4_port_type_e port_type) 330 { 331 uint32_t i; 332 char prop_buf[32]; 333 334 if (hw == NULL) { 335 ocs_log_err(os, "bad parameter(s) hw=%p\n", hw); 336 return OCS_HW_RTN_ERROR; 337 } 338 339 if (hw->hw_setup_called) { 340 /* Setup run-time workarounds. 341 * Call for each setup, to allow for hw_war_version 342 */ 343 ocs_hw_workaround_setup(hw); 344 return OCS_HW_RTN_SUCCESS; 345 } 346 347 /* 348 * ocs_hw_init() relies on NULL pointers indicating that a structure 349 * needs allocation. If a structure is non-NULL, ocs_hw_init() won't 350 * free/realloc that memory 351 */ 352 ocs_memset(hw, 0, sizeof(ocs_hw_t)); 353 354 hw->hw_setup_called = TRUE; 355 356 hw->os = os; 357 358 ocs_lock_init(hw->os, &hw->cmd_lock, "HW_cmd_lock[%d]", ocs_instance(hw->os)); 359 ocs_list_init(&hw->cmd_head, ocs_command_ctx_t, link); 360 ocs_list_init(&hw->cmd_pending, ocs_command_ctx_t, link); 361 hw->cmd_head_count = 0; 362 363 ocs_lock_init(hw->os, &hw->io_lock, "HW_io_lock[%d]", ocs_instance(hw->os)); 364 ocs_lock_init(hw->os, &hw->io_abort_lock, "HW_io_abort_lock[%d]", ocs_instance(hw->os)); 365 366 ocs_atomic_init(&hw->io_alloc_failed_count, 0); 367 368 hw->config.speed = FC_LINK_SPEED_AUTO_16_8_4; 369 hw->config.dif_seed = 0; 370 hw->config.auto_xfer_rdy_blk_size_chip = OCS_HW_AUTO_XFER_RDY_BLK_SIZE_DEFAULT; 371 hw->config.auto_xfer_rdy_ref_tag_is_lba = OCS_HW_AUTO_XFER_RDY_REF_TAG_IS_LBA_DEFAULT; 372 hw->config.auto_xfer_rdy_app_tag_valid = OCS_HW_AUTO_XFER_RDY_APP_TAG_VALID_DEFAULT; 373 hw->config.auto_xfer_rdy_app_tag_value = OCS_HW_AUTO_XFER_RDY_APP_TAG_VALUE_DEFAULT; 374 375 if (sli_setup(&hw->sli, hw->os, port_type)) { 376 ocs_log_err(hw->os, "SLI setup failed\n"); 377 return OCS_HW_RTN_ERROR; 378 } 379 380 ocs_memset(hw->domains, 0, sizeof(hw->domains)); 381 382 ocs_memset(hw->fcf_index_fcfi, 0, sizeof(hw->fcf_index_fcfi)); 383 384 ocs_hw_link_event_init(hw); 385 386 sli_callback(&hw->sli, SLI4_CB_LINK, ocs_hw_cb_link, hw); 387 sli_callback(&hw->sli, SLI4_CB_FIP, ocs_hw_cb_fip, hw); 388 389 /* 390 * Set all the queue sizes to the maximum allowed. These values may 391 * be changes later by the adjust and workaround functions. 392 */ 393 for (i = 0; i < ARRAY_SIZE(hw->num_qentries); i++) { 394 hw->num_qentries[i] = sli_get_max_qentries(&hw->sli, i); 395 } 396 397 /* 398 * The RQ assignment for RQ pair mode. 399 */ 400 hw->config.rq_default_buffer_size = OCS_HW_RQ_SIZE_PAYLOAD; 401 hw->config.n_io = sli_get_max_rsrc(&hw->sli, SLI_RSRC_FCOE_XRI); 402 if (ocs_get_property("auto_xfer_rdy_xri_cnt", prop_buf, sizeof(prop_buf)) == 0) { 403 hw->config.auto_xfer_rdy_xri_cnt = ocs_strtoul(prop_buf, 0, 0); 404 } 405 406 /* by default, enable initiator-only auto-ABTS emulation */ 407 hw->config.i_only_aab = TRUE; 408 409 /* Setup run-time workarounds */ 410 ocs_hw_workaround_setup(hw); 411 412 /* HW_WORKAROUND_OVERRIDE_FCFI_IN_SRB */ 413 if (hw->workaround.override_fcfi) { 414 hw->first_domain_idx = -1; 415 } 416 417 /* Must be done after the workaround setup */ 418 if ((SLI4_IF_TYPE_LANCER_FC_ETH == sli_get_if_type(&hw->sli)) || 419 (SLI4_IF_TYPE_LANCER_G7 == sli_get_if_type(&hw->sli))) { 420 421 (void)ocs_hw_read_max_dump_size(hw); 422 } 423 424 /* calculate the number of WQs required. */ 425 ocs_hw_adjust_wqs(hw); 426 427 /* Set the default dif mode */ 428 if (! sli_is_dif_inline_capable(&hw->sli)) { 429 ocs_log_test(hw->os, "not inline capable, setting mode to separate\n"); 430 hw->config.dif_mode = OCS_HW_DIF_MODE_SEPARATE; 431 } 432 /* Workaround: BZ 161832 */ 433 if (hw->workaround.use_dif_sec_xri) { 434 ocs_list_init(&hw->sec_hio_wait_list, ocs_hw_io_t, link); 435 } 436 437 /* 438 * Figure out the starting and max ULP to spread the WQs across the 439 * ULPs. 440 */ 441 if (sli_get_is_dual_ulp_capable(&hw->sli)) { 442 if (sli_get_is_ulp_enabled(&hw->sli, 0) && 443 sli_get_is_ulp_enabled(&hw->sli, 1)) { 444 hw->ulp_start = 0; 445 hw->ulp_max = 1; 446 } else if (sli_get_is_ulp_enabled(&hw->sli, 0)) { 447 hw->ulp_start = 0; 448 hw->ulp_max = 0; 449 } else { 450 hw->ulp_start = 1; 451 hw->ulp_max = 1; 452 } 453 } else { 454 if (sli_get_is_ulp_enabled(&hw->sli, 0)) { 455 hw->ulp_start = 0; 456 hw->ulp_max = 0; 457 } else { 458 hw->ulp_start = 1; 459 hw->ulp_max = 1; 460 } 461 } 462 ocs_log_debug(hw->os, "ulp_start %d, ulp_max %d\n", 463 hw->ulp_start, hw->ulp_max); 464 hw->config.queue_topology = hw_global.queue_topology_string; 465 466 hw->qtop = ocs_hw_qtop_parse(hw, hw->config.queue_topology); 467 468 hw->config.n_eq = hw->qtop->entry_counts[QTOP_EQ]; 469 hw->config.n_cq = hw->qtop->entry_counts[QTOP_CQ]; 470 hw->config.n_rq = hw->qtop->entry_counts[QTOP_RQ]; 471 hw->config.n_wq = hw->qtop->entry_counts[QTOP_WQ]; 472 hw->config.n_mq = hw->qtop->entry_counts[QTOP_MQ]; 473 474 /* Verify qtop configuration against driver supported configuration */ 475 if (hw->config.n_rq > OCE_HW_MAX_NUM_MRQ_PAIRS) { 476 ocs_log_crit(hw->os, "Max supported MRQ pairs = %d\n", 477 OCE_HW_MAX_NUM_MRQ_PAIRS); 478 return OCS_HW_RTN_ERROR; 479 } 480 481 if (hw->config.n_eq > OCS_HW_MAX_NUM_EQ) { 482 ocs_log_crit(hw->os, "Max supported EQs = %d\n", 483 OCS_HW_MAX_NUM_EQ); 484 return OCS_HW_RTN_ERROR; 485 } 486 487 if (hw->config.n_cq > OCS_HW_MAX_NUM_CQ) { 488 ocs_log_crit(hw->os, "Max supported CQs = %d\n", 489 OCS_HW_MAX_NUM_CQ); 490 return OCS_HW_RTN_ERROR; 491 } 492 493 if (hw->config.n_wq > OCS_HW_MAX_NUM_WQ) { 494 ocs_log_crit(hw->os, "Max supported WQs = %d\n", 495 OCS_HW_MAX_NUM_WQ); 496 return OCS_HW_RTN_ERROR; 497 } 498 499 if (hw->config.n_mq > OCS_HW_MAX_NUM_MQ) { 500 ocs_log_crit(hw->os, "Max supported MQs = %d\n", 501 OCS_HW_MAX_NUM_MQ); 502 return OCS_HW_RTN_ERROR; 503 } 504 505 return OCS_HW_RTN_SUCCESS; 506 } 507 508 /** 509 * @ingroup devInitShutdown 510 * @brief Allocate memory structures to prepare for the device operation. 511 * 512 * @par Description 513 * Allocates memory structures needed by the device and prepares the device 514 * for operation. 515 * @n @n @b Note: This function may be called more than once (for example, at 516 * initialization and then after a reset), but the size of the internal resources 517 * may not be changed without tearing down the HW (ocs_hw_teardown()). 518 * 519 * @param hw Hardware context allocated by the caller. 520 * 521 * @return Returns 0 on success, or a non-zero value on failure. 522 */ 523 ocs_hw_rtn_e 524 ocs_hw_init(ocs_hw_t *hw) 525 { 526 ocs_hw_rtn_e rc; 527 uint32_t i = 0; 528 uint8_t buf[SLI4_BMBX_SIZE]; 529 uint32_t max_rpi; 530 int rem_count; 531 int written_size = 0; 532 uint32_t count; 533 char prop_buf[32]; 534 uint32_t ramdisc_blocksize = 512; 535 uint32_t q_count = 0; 536 /* 537 * Make sure the command lists are empty. If this is start-of-day, 538 * they'll be empty since they were just initialized in ocs_hw_setup. 539 * If we've just gone through a reset, the command and command pending 540 * lists should have been cleaned up as part of the reset (ocs_hw_reset()). 541 */ 542 ocs_lock(&hw->cmd_lock); 543 if (!ocs_list_empty(&hw->cmd_head)) { 544 ocs_log_test(hw->os, "command found on cmd list\n"); 545 ocs_unlock(&hw->cmd_lock); 546 return OCS_HW_RTN_ERROR; 547 } 548 if (!ocs_list_empty(&hw->cmd_pending)) { 549 ocs_log_test(hw->os, "command found on pending list\n"); 550 ocs_unlock(&hw->cmd_lock); 551 return OCS_HW_RTN_ERROR; 552 } 553 ocs_unlock(&hw->cmd_lock); 554 555 /* Free RQ buffers if prevously allocated */ 556 ocs_hw_rx_free(hw); 557 558 /* 559 * The IO queues must be initialized here for the reset case. The 560 * ocs_hw_init_io() function will re-add the IOs to the free list. 561 * The cmd_head list should be OK since we free all entries in 562 * ocs_hw_command_cancel() that is called in the ocs_hw_reset(). 563 */ 564 565 /* If we are in this function due to a reset, there may be stale items 566 * on lists that need to be removed. Clean them up. 567 */ 568 rem_count=0; 569 if (ocs_list_valid(&hw->io_wait_free)) { 570 while ((!ocs_list_empty(&hw->io_wait_free))) { 571 rem_count++; 572 ocs_list_remove_head(&hw->io_wait_free); 573 } 574 if (rem_count > 0) { 575 ocs_log_debug(hw->os, "removed %d items from io_wait_free list\n", rem_count); 576 } 577 } 578 rem_count=0; 579 if (ocs_list_valid(&hw->io_inuse)) { 580 while ((!ocs_list_empty(&hw->io_inuse))) { 581 rem_count++; 582 ocs_list_remove_head(&hw->io_inuse); 583 } 584 if (rem_count > 0) { 585 ocs_log_debug(hw->os, "removed %d items from io_inuse list\n", rem_count); 586 } 587 } 588 rem_count=0; 589 if (ocs_list_valid(&hw->io_free)) { 590 while ((!ocs_list_empty(&hw->io_free))) { 591 rem_count++; 592 ocs_list_remove_head(&hw->io_free); 593 } 594 if (rem_count > 0) { 595 ocs_log_debug(hw->os, "removed %d items from io_free list\n", rem_count); 596 } 597 } 598 if (ocs_list_valid(&hw->io_port_owned)) { 599 while ((!ocs_list_empty(&hw->io_port_owned))) { 600 ocs_list_remove_head(&hw->io_port_owned); 601 } 602 } 603 ocs_list_init(&hw->io_inuse, ocs_hw_io_t, link); 604 ocs_list_init(&hw->io_free, ocs_hw_io_t, link); 605 ocs_list_init(&hw->io_port_owned, ocs_hw_io_t, link); 606 ocs_list_init(&hw->io_wait_free, ocs_hw_io_t, link); 607 ocs_list_init(&hw->io_timed_wqe, ocs_hw_io_t, wqe_link); 608 ocs_list_init(&hw->io_port_dnrx, ocs_hw_io_t, dnrx_link); 609 610 /* If MRQ not required, Make sure we dont request feature. */ 611 if (hw->config.n_rq == 1) { 612 hw->sli.config.features.flag.mrqp = FALSE; 613 } 614 615 if (sli_init(&hw->sli)) { 616 ocs_log_err(hw->os, "SLI failed to initialize\n"); 617 return OCS_HW_RTN_ERROR; 618 } 619 620 /* 621 * Enable the auto xfer rdy feature if requested. 622 */ 623 hw->auto_xfer_rdy_enabled = FALSE; 624 if (sli_get_auto_xfer_rdy_capable(&hw->sli) && 625 hw->config.auto_xfer_rdy_size > 0) { 626 if (hw->config.esoc){ 627 if (ocs_get_property("ramdisc_blocksize", prop_buf, sizeof(prop_buf)) == 0) { 628 ramdisc_blocksize = ocs_strtoul(prop_buf, 0, 0); 629 } 630 written_size = sli_cmd_config_auto_xfer_rdy_hp(&hw->sli, buf, SLI4_BMBX_SIZE, hw->config.auto_xfer_rdy_size, 1, ramdisc_blocksize); 631 } else { 632 written_size = sli_cmd_config_auto_xfer_rdy(&hw->sli, buf, SLI4_BMBX_SIZE, hw->config.auto_xfer_rdy_size); 633 } 634 if (written_size) { 635 rc = ocs_hw_command(hw, buf, OCS_CMD_POLL, NULL, NULL); 636 if (rc != OCS_HW_RTN_SUCCESS) { 637 ocs_log_err(hw->os, "config auto xfer rdy failed\n"); 638 return rc; 639 } 640 } 641 hw->auto_xfer_rdy_enabled = TRUE; 642 643 if (hw->config.auto_xfer_rdy_t10_enable) { 644 rc = ocs_hw_config_auto_xfer_rdy_t10pi(hw, buf); 645 if (rc != OCS_HW_RTN_SUCCESS) { 646 ocs_log_err(hw->os, "set parameters auto xfer rdy T10 PI failed\n"); 647 return rc; 648 } 649 } 650 } 651 652 if(hw->sliport_healthcheck) { 653 rc = ocs_hw_config_sli_port_health_check(hw, 0, 1); 654 if (rc != OCS_HW_RTN_SUCCESS) { 655 ocs_log_err(hw->os, "Enabling Sliport Health check failed \n"); 656 return rc; 657 } 658 } 659 660 /* 661 * Set FDT transfer hint, only works on Lancer 662 */ 663 if ((hw->sli.if_type == SLI4_IF_TYPE_LANCER_FC_ETH) && (OCS_HW_FDT_XFER_HINT != 0)) { 664 /* 665 * Non-fatal error. In particular, we can disregard failure to set OCS_HW_FDT_XFER_HINT on 666 * devices with legacy firmware that do not support OCS_HW_FDT_XFER_HINT feature. 667 */ 668 ocs_hw_config_set_fdt_xfer_hint(hw, OCS_HW_FDT_XFER_HINT); 669 } 670 671 /* 672 * Verify that we have not exceeded any queue sizes 673 */ 674 q_count = MIN(sli_get_max_queue(&hw->sli, SLI_QTYPE_EQ), 675 OCS_HW_MAX_NUM_EQ); 676 if (hw->config.n_eq > q_count) { 677 ocs_log_err(hw->os, "requested %d EQ but %d allowed\n", 678 hw->config.n_eq, q_count); 679 return OCS_HW_RTN_ERROR; 680 } 681 682 q_count = MIN(sli_get_max_queue(&hw->sli, SLI_QTYPE_CQ), 683 OCS_HW_MAX_NUM_CQ); 684 if (hw->config.n_cq > q_count) { 685 ocs_log_err(hw->os, "requested %d CQ but %d allowed\n", 686 hw->config.n_cq, q_count); 687 return OCS_HW_RTN_ERROR; 688 } 689 690 q_count = MIN(sli_get_max_queue(&hw->sli, SLI_QTYPE_MQ), 691 OCS_HW_MAX_NUM_MQ); 692 if (hw->config.n_mq > q_count) { 693 ocs_log_err(hw->os, "requested %d MQ but %d allowed\n", 694 hw->config.n_mq, q_count); 695 return OCS_HW_RTN_ERROR; 696 } 697 698 q_count = MIN(sli_get_max_queue(&hw->sli, SLI_QTYPE_RQ), 699 OCS_HW_MAX_NUM_RQ); 700 if (hw->config.n_rq > q_count) { 701 ocs_log_err(hw->os, "requested %d RQ but %d allowed\n", 702 hw->config.n_rq, q_count); 703 return OCS_HW_RTN_ERROR; 704 } 705 706 q_count = MIN(sli_get_max_queue(&hw->sli, SLI_QTYPE_WQ), 707 OCS_HW_MAX_NUM_WQ); 708 if (hw->config.n_wq > q_count) { 709 ocs_log_err(hw->os, "requested %d WQ but %d allowed\n", 710 hw->config.n_wq, q_count); 711 return OCS_HW_RTN_ERROR; 712 } 713 714 /* zero the hashes */ 715 ocs_memset(hw->cq_hash, 0, sizeof(hw->cq_hash)); 716 ocs_log_debug(hw->os, "Max CQs %d, hash size = %d\n", 717 OCS_HW_MAX_NUM_CQ, OCS_HW_Q_HASH_SIZE); 718 719 ocs_memset(hw->rq_hash, 0, sizeof(hw->rq_hash)); 720 ocs_log_debug(hw->os, "Max RQs %d, hash size = %d\n", 721 OCS_HW_MAX_NUM_RQ, OCS_HW_Q_HASH_SIZE); 722 723 ocs_memset(hw->wq_hash, 0, sizeof(hw->wq_hash)); 724 ocs_log_debug(hw->os, "Max WQs %d, hash size = %d\n", 725 OCS_HW_MAX_NUM_WQ, OCS_HW_Q_HASH_SIZE); 726 727 rc = ocs_hw_init_queues(hw, hw->qtop); 728 if (rc != OCS_HW_RTN_SUCCESS) { 729 return rc; 730 } 731 732 max_rpi = sli_get_max_rsrc(&hw->sli, SLI_RSRC_FCOE_RPI); 733 i = sli_fc_get_rpi_requirements(&hw->sli, max_rpi); 734 if (i) { 735 ocs_dma_t payload_memory; 736 737 rc = OCS_HW_RTN_ERROR; 738 739 if (hw->rnode_mem.size) { 740 ocs_dma_free(hw->os, &hw->rnode_mem); 741 } 742 743 if (ocs_dma_alloc(hw->os, &hw->rnode_mem, i, 4096)) { 744 ocs_log_err(hw->os, "remote node memory allocation fail\n"); 745 return OCS_HW_RTN_NO_MEMORY; 746 } 747 748 payload_memory.size = 0; 749 if (sli_cmd_fcoe_post_hdr_templates(&hw->sli, buf, SLI4_BMBX_SIZE, 750 &hw->rnode_mem, UINT16_MAX, &payload_memory)) { 751 rc = ocs_hw_command(hw, buf, OCS_CMD_POLL, NULL, NULL); 752 753 if (payload_memory.size != 0) { 754 /* The command was non-embedded - need to free the dma buffer */ 755 ocs_dma_free(hw->os, &payload_memory); 756 } 757 } 758 759 if (rc != OCS_HW_RTN_SUCCESS) { 760 ocs_log_err(hw->os, "header template registration failed\n"); 761 return rc; 762 } 763 } 764 765 /* Allocate and post RQ buffers */ 766 rc = ocs_hw_rx_allocate(hw); 767 if (rc) { 768 ocs_log_err(hw->os, "rx_allocate failed\n"); 769 return rc; 770 } 771 772 /* Populate hw->seq_free_list */ 773 if (hw->seq_pool == NULL) { 774 uint32_t count = 0; 775 uint32_t i; 776 777 /* Sum up the total number of RQ entries, to use to allocate the sequence object pool */ 778 for (i = 0; i < hw->hw_rq_count; i++) { 779 count += hw->hw_rq[i]->entry_count; 780 } 781 782 hw->seq_pool = ocs_array_alloc(hw->os, sizeof(ocs_hw_sequence_t), count); 783 if (hw->seq_pool == NULL) { 784 ocs_log_err(hw->os, "malloc seq_pool failed\n"); 785 return OCS_HW_RTN_NO_MEMORY; 786 } 787 } 788 789 if(ocs_hw_rx_post(hw)) { 790 ocs_log_err(hw->os, "WARNING - error posting RQ buffers\n"); 791 } 792 793 /* Allocate rpi_ref if not previously allocated */ 794 if (hw->rpi_ref == NULL) { 795 hw->rpi_ref = ocs_malloc(hw->os, max_rpi * sizeof(*hw->rpi_ref), 796 OCS_M_ZERO | OCS_M_NOWAIT); 797 if (hw->rpi_ref == NULL) { 798 ocs_log_err(hw->os, "rpi_ref allocation failure (%d)\n", i); 799 return OCS_HW_RTN_NO_MEMORY; 800 } 801 } 802 803 for (i = 0; i < max_rpi; i ++) { 804 ocs_atomic_init(&hw->rpi_ref[i].rpi_count, 0); 805 ocs_atomic_init(&hw->rpi_ref[i].rpi_attached, 0); 806 } 807 808 ocs_memset(hw->domains, 0, sizeof(hw->domains)); 809 810 /* HW_WORKAROUND_OVERRIDE_FCFI_IN_SRB */ 811 if (hw->workaround.override_fcfi) { 812 hw->first_domain_idx = -1; 813 } 814 815 ocs_memset(hw->fcf_index_fcfi, 0, sizeof(hw->fcf_index_fcfi)); 816 817 /* Register a FCFI to allow unsolicited frames to be routed to the driver */ 818 if (sli_get_medium(&hw->sli) == SLI_LINK_MEDIUM_FC) { 819 if (hw->hw_mrq_count) { 820 ocs_log_debug(hw->os, "using REG_FCFI MRQ\n"); 821 822 rc = ocs_hw_config_mrq(hw, SLI4_CMD_REG_FCFI_SET_FCFI_MODE, 0, 0); 823 if (rc != OCS_HW_RTN_SUCCESS) { 824 ocs_log_err(hw->os, "REG_FCFI_MRQ FCFI registration failed\n"); 825 return rc; 826 } 827 828 rc = ocs_hw_config_mrq(hw, SLI4_CMD_REG_FCFI_SET_MRQ_MODE, 0, 0); 829 if (rc != OCS_HW_RTN_SUCCESS) { 830 ocs_log_err(hw->os, "REG_FCFI_MRQ MRQ registration failed\n"); 831 return rc; 832 } 833 } else { 834 sli4_cmd_rq_cfg_t rq_cfg[SLI4_CMD_REG_FCFI_NUM_RQ_CFG]; 835 836 ocs_log_debug(hw->os, "using REG_FCFI standard\n"); 837 838 /* Set the filter match/mask values from hw's filter_def values */ 839 for (i = 0; i < SLI4_CMD_REG_FCFI_NUM_RQ_CFG; i++) { 840 rq_cfg[i].rq_id = 0xffff; 841 rq_cfg[i].r_ctl_mask = (uint8_t) hw->config.filter_def[i]; 842 rq_cfg[i].r_ctl_match = (uint8_t) (hw->config.filter_def[i] >> 8); 843 rq_cfg[i].type_mask = (uint8_t) (hw->config.filter_def[i] >> 16); 844 rq_cfg[i].type_match = (uint8_t) (hw->config.filter_def[i] >> 24); 845 } 846 847 /* 848 * Update the rq_id's of the FCF configuration (don't update more than the number 849 * of rq_cfg elements) 850 */ 851 for (i = 0; i < OCS_MIN(hw->hw_rq_count, SLI4_CMD_REG_FCFI_NUM_RQ_CFG); i++) { 852 hw_rq_t *rq = hw->hw_rq[i]; 853 uint32_t j; 854 for (j = 0; j < SLI4_CMD_REG_FCFI_NUM_RQ_CFG; j++) { 855 uint32_t mask = (rq->filter_mask != 0) ? rq->filter_mask : 1; 856 if (mask & (1U << j)) { 857 rq_cfg[j].rq_id = rq->hdr->id; 858 ocs_log_debug(hw->os, "REG_FCFI: filter[%d] %08X -> RQ[%d] id=%d\n", 859 j, hw->config.filter_def[j], i, rq->hdr->id); 860 } 861 } 862 } 863 864 rc = OCS_HW_RTN_ERROR; 865 866 if (sli_cmd_reg_fcfi(&hw->sli, buf, SLI4_BMBX_SIZE, 0, rq_cfg, 0)) { 867 rc = ocs_hw_command(hw, buf, OCS_CMD_POLL, NULL, NULL); 868 } 869 870 if (rc != OCS_HW_RTN_SUCCESS) { 871 ocs_log_err(hw->os, "FCFI registration failed\n"); 872 return rc; 873 } 874 hw->fcf_indicator = ((sli4_cmd_reg_fcfi_t *)buf)->fcfi; 875 } 876 } 877 878 /* 879 * Allocate the WQ request tag pool, if not previously allocated (the request tag value is 16 bits, 880 * thus the pool allocation size of 64k) 881 */ 882 rc = ocs_hw_reqtag_init(hw); 883 if (rc) { 884 ocs_log_err(hw->os, "ocs_pool_alloc hw_wq_callback_t failed: %d\n", rc); 885 return rc; 886 } 887 888 rc = ocs_hw_setup_io(hw); 889 if (rc) { 890 ocs_log_err(hw->os, "IO allocation failure\n"); 891 return rc; 892 } 893 894 rc = ocs_hw_init_io(hw); 895 if (rc) { 896 ocs_log_err(hw->os, "IO initialization failure\n"); 897 return rc; 898 } 899 900 ocs_queue_history_init(hw->os, &hw->q_hist); 901 902 /* get hw link config; polling, so callback will be called immediately */ 903 hw->linkcfg = OCS_HW_LINKCFG_NA; 904 ocs_hw_get_linkcfg(hw, OCS_CMD_POLL, ocs_hw_init_linkcfg_cb, hw); 905 906 /* if lancer ethernet, ethernet ports need to be enabled */ 907 if ((hw->sli.if_type == SLI4_IF_TYPE_LANCER_FC_ETH) && 908 (sli_get_medium(&hw->sli) == SLI_LINK_MEDIUM_ETHERNET)) { 909 if (ocs_hw_set_eth_license(hw, hw->eth_license)) { 910 /* log warning but continue */ 911 ocs_log_err(hw->os, "Failed to set ethernet license\n"); 912 } 913 } 914 915 /* Set the DIF seed - only for lancer right now */ 916 if (SLI4_IF_TYPE_LANCER_FC_ETH == sli_get_if_type(&hw->sli) && 917 ocs_hw_set_dif_seed(hw) != OCS_HW_RTN_SUCCESS) { 918 ocs_log_err(hw->os, "Failed to set DIF seed value\n"); 919 return rc; 920 } 921 922 /* Set the DIF mode - skyhawk only */ 923 if (SLI4_IF_TYPE_BE3_SKH_PF == sli_get_if_type(&hw->sli) && 924 sli_get_dif_capable(&hw->sli)) { 925 rc = ocs_hw_set_dif_mode(hw); 926 if (rc != OCS_HW_RTN_SUCCESS) { 927 ocs_log_err(hw->os, "Failed to set DIF mode value\n"); 928 return rc; 929 } 930 } 931 932 /* 933 * Arming the EQ allows (e.g.) interrupts when CQ completions write EQ entries 934 */ 935 for (i = 0; i < hw->eq_count; i++) { 936 sli_queue_arm(&hw->sli, &hw->eq[i], TRUE); 937 } 938 939 /* 940 * Initialize RQ hash 941 */ 942 for (i = 0; i < hw->rq_count; i++) { 943 ocs_hw_queue_hash_add(hw->rq_hash, hw->rq[i].id, i); 944 } 945 946 /* 947 * Initialize WQ hash 948 */ 949 for (i = 0; i < hw->wq_count; i++) { 950 ocs_hw_queue_hash_add(hw->wq_hash, hw->wq[i].id, i); 951 } 952 953 /* 954 * Arming the CQ allows (e.g.) MQ completions to write CQ entries 955 */ 956 for (i = 0; i < hw->cq_count; i++) { 957 ocs_hw_queue_hash_add(hw->cq_hash, hw->cq[i].id, i); 958 sli_queue_arm(&hw->sli, &hw->cq[i], TRUE); 959 } 960 961 /* record the fact that the queues are functional */ 962 hw->state = OCS_HW_STATE_ACTIVE; 963 964 /* Note: Must be after the IOs are setup and the state is active*/ 965 if (ocs_hw_rqpair_init(hw)) { 966 ocs_log_err(hw->os, "WARNING - error initializing RQ pair\n"); 967 } 968 969 /* finally kick off periodic timer to check for timed out target WQEs */ 970 if (hw->config.emulate_tgt_wqe_timeout) { 971 ocs_setup_timer(hw->os, &hw->wqe_timer, target_wqe_timer_cb, hw, 972 OCS_HW_WQ_TIMER_PERIOD_MS); 973 } 974 975 /* 976 * Allocate a HW IOs for send frame. Allocate one for each Class 1 WQ, or if there 977 * are none of those, allocate one for WQ[0] 978 */ 979 if ((count = ocs_varray_get_count(hw->wq_class_array[1])) > 0) { 980 for (i = 0; i < count; i++) { 981 hw_wq_t *wq = ocs_varray_iter_next(hw->wq_class_array[1]); 982 wq->send_frame_io = ocs_hw_io_alloc(hw); 983 if (wq->send_frame_io == NULL) { 984 ocs_log_err(hw->os, "ocs_hw_io_alloc for send_frame_io failed\n"); 985 } 986 } 987 } else { 988 hw->hw_wq[0]->send_frame_io = ocs_hw_io_alloc(hw); 989 if (hw->hw_wq[0]->send_frame_io == NULL) { 990 ocs_log_err(hw->os, "ocs_hw_io_alloc for send_frame_io failed\n"); 991 } 992 } 993 994 /* Initialize send frame frame sequence id */ 995 ocs_atomic_init(&hw->send_frame_seq_id, 0); 996 997 /* Initialize watchdog timer if enabled by user */ 998 hw->expiration_logged = 0; 999 if(hw->watchdog_timeout) { 1000 if((hw->watchdog_timeout < 1) || (hw->watchdog_timeout > 65534)) { 1001 ocs_log_err(hw->os, "watchdog_timeout out of range: Valid range is 1 - 65534\n"); 1002 }else if(!ocs_hw_config_watchdog_timer(hw)) { 1003 ocs_log_info(hw->os, "watchdog timer configured with timeout = %d seconds \n", hw->watchdog_timeout); 1004 } 1005 } 1006 1007 if (ocs_dma_alloc(hw->os, &hw->domain_dmem, 112, 4)) { 1008 ocs_log_err(hw->os, "domain node memory allocation fail\n"); 1009 return OCS_HW_RTN_NO_MEMORY; 1010 } 1011 1012 if (ocs_dma_alloc(hw->os, &hw->fcf_dmem, OCS_HW_READ_FCF_SIZE, OCS_HW_READ_FCF_SIZE)) { 1013 ocs_log_err(hw->os, "domain fcf memory allocation fail\n"); 1014 return OCS_HW_RTN_NO_MEMORY; 1015 } 1016 1017 if ((0 == hw->loop_map.size) && ocs_dma_alloc(hw->os, &hw->loop_map, 1018 SLI4_MIN_LOOP_MAP_BYTES, 4)) { 1019 ocs_log_err(hw->os, "Loop dma alloc failed size:%d \n", hw->loop_map.size); 1020 } 1021 1022 return OCS_HW_RTN_SUCCESS; 1023 } 1024 1025 /** 1026 * @brief Configure Multi-RQ 1027 * 1028 * @param hw Hardware context allocated by the caller. 1029 * @param mode 1 to set MRQ filters and 0 to set FCFI index 1030 * @param vlanid valid in mode 0 1031 * @param fcf_index valid in mode 0 1032 * 1033 * @return Returns 0 on success, or a non-zero value on failure. 1034 */ 1035 static int32_t 1036 ocs_hw_config_mrq(ocs_hw_t *hw, uint8_t mode, uint16_t vlanid, uint16_t fcf_index) 1037 { 1038 uint8_t buf[SLI4_BMBX_SIZE], mrq_bitmask = 0; 1039 hw_rq_t *rq; 1040 sli4_cmd_reg_fcfi_mrq_t *rsp = NULL; 1041 uint32_t i, j; 1042 sli4_cmd_rq_cfg_t rq_filter[SLI4_CMD_REG_FCFI_MRQ_NUM_RQ_CFG]; 1043 int32_t rc; 1044 1045 if (mode == SLI4_CMD_REG_FCFI_SET_FCFI_MODE) { 1046 goto issue_cmd; 1047 } 1048 1049 /* Set the filter match/mask values from hw's filter_def values */ 1050 for (i = 0; i < SLI4_CMD_REG_FCFI_NUM_RQ_CFG; i++) { 1051 rq_filter[i].rq_id = 0xffff; 1052 rq_filter[i].r_ctl_mask = (uint8_t) hw->config.filter_def[i]; 1053 rq_filter[i].r_ctl_match = (uint8_t) (hw->config.filter_def[i] >> 8); 1054 rq_filter[i].type_mask = (uint8_t) (hw->config.filter_def[i] >> 16); 1055 rq_filter[i].type_match = (uint8_t) (hw->config.filter_def[i] >> 24); 1056 } 1057 1058 /* Accumulate counts for each filter type used, build rq_ids[] list */ 1059 for (i = 0; i < hw->hw_rq_count; i++) { 1060 rq = hw->hw_rq[i]; 1061 for (j = 0; j < SLI4_CMD_REG_FCFI_MRQ_NUM_RQ_CFG; j++) { 1062 if (rq->filter_mask & (1U << j)) { 1063 if (rq_filter[j].rq_id != 0xffff) { 1064 /* Already used. Bailout ifts not RQset case */ 1065 if (!rq->is_mrq || (rq_filter[j].rq_id != rq->base_mrq_id)) { 1066 ocs_log_err(hw->os, "Wrong queue topology.\n"); 1067 return OCS_HW_RTN_ERROR; 1068 } 1069 continue; 1070 } 1071 1072 if (rq->is_mrq) { 1073 rq_filter[j].rq_id = rq->base_mrq_id; 1074 mrq_bitmask |= (1U << j); 1075 } else { 1076 rq_filter[j].rq_id = rq->hdr->id; 1077 } 1078 } 1079 } 1080 } 1081 1082 issue_cmd: 1083 /* Invoke REG_FCFI_MRQ */ 1084 rc = sli_cmd_reg_fcfi_mrq(&hw->sli, 1085 buf, /* buf */ 1086 SLI4_BMBX_SIZE, /* size */ 1087 mode, /* mode 1 */ 1088 fcf_index, /* fcf_index */ 1089 vlanid, /* vlan_id */ 1090 hw->config.rq_selection_policy, /* RQ selection policy*/ 1091 mrq_bitmask, /* MRQ bitmask */ 1092 hw->hw_mrq_count, /* num_mrqs */ 1093 rq_filter); /* RQ filter */ 1094 if (rc == 0) { 1095 ocs_log_err(hw->os, "sli_cmd_reg_fcfi_mrq() failed: %d\n", rc); 1096 return OCS_HW_RTN_ERROR; 1097 } 1098 1099 rc = ocs_hw_command(hw, buf, OCS_CMD_POLL, NULL, NULL); 1100 1101 rsp = (sli4_cmd_reg_fcfi_mrq_t *)buf; 1102 1103 if ((rc != OCS_HW_RTN_SUCCESS) || (rsp->hdr.status)) { 1104 ocs_log_err(hw->os, "FCFI MRQ registration failed. cmd = %x status = %x\n", 1105 rsp->hdr.command, rsp->hdr.status); 1106 return OCS_HW_RTN_ERROR; 1107 } 1108 1109 if (mode == SLI4_CMD_REG_FCFI_SET_FCFI_MODE) { 1110 hw->fcf_indicator = rsp->fcfi; 1111 } 1112 return 0; 1113 } 1114 1115 /** 1116 * @brief Callback function for getting linkcfg during HW initialization. 1117 * 1118 * @param status Status of the linkcfg get operation. 1119 * @param value Link configuration enum to which the link configuration is set. 1120 * @param arg Callback argument (ocs_hw_t *). 1121 * 1122 * @return None. 1123 */ 1124 static void 1125 ocs_hw_init_linkcfg_cb(int32_t status, uintptr_t value, void *arg) 1126 { 1127 ocs_hw_t *hw = (ocs_hw_t *)arg; 1128 if (status == 0) { 1129 hw->linkcfg = (ocs_hw_linkcfg_e)value; 1130 } else { 1131 hw->linkcfg = OCS_HW_LINKCFG_NA; 1132 } 1133 ocs_log_debug(hw->os, "linkcfg=%d\n", hw->linkcfg); 1134 } 1135 1136 /** 1137 * @ingroup devInitShutdown 1138 * @brief Tear down the Hardware Abstraction Layer module. 1139 * 1140 * @par Description 1141 * Frees memory structures needed by the device, and shuts down the device. Does 1142 * not free the HW context memory (which is done by the caller). 1143 * 1144 * @param hw Hardware context allocated by the caller. 1145 * 1146 * @return Returns 0 on success, or a non-zero value on failure. 1147 */ 1148 ocs_hw_rtn_e 1149 ocs_hw_teardown(ocs_hw_t *hw) 1150 { 1151 uint32_t i = 0; 1152 uint32_t iters = 10;/*XXX*/ 1153 uint32_t max_rpi; 1154 uint32_t destroy_queues; 1155 uint32_t free_memory; 1156 1157 if (!hw) { 1158 ocs_log_err(NULL, "bad parameter(s) hw=%p\n", hw); 1159 return OCS_HW_RTN_ERROR; 1160 } 1161 1162 destroy_queues = (hw->state == OCS_HW_STATE_ACTIVE); 1163 free_memory = (hw->state != OCS_HW_STATE_UNINITIALIZED); 1164 1165 /* shutdown target wqe timer */ 1166 shutdown_target_wqe_timer(hw); 1167 1168 /* Cancel watchdog timer if enabled */ 1169 if(hw->watchdog_timeout) { 1170 hw->watchdog_timeout = 0; 1171 ocs_hw_config_watchdog_timer(hw); 1172 } 1173 1174 /* Cancel Sliport Healthcheck */ 1175 if(hw->sliport_healthcheck) { 1176 hw->sliport_healthcheck = 0; 1177 ocs_hw_config_sli_port_health_check(hw, 0, 0); 1178 } 1179 1180 if (hw->state != OCS_HW_STATE_QUEUES_ALLOCATED) { 1181 hw->state = OCS_HW_STATE_TEARDOWN_IN_PROGRESS; 1182 1183 ocs_hw_flush(hw); 1184 1185 /* If there are outstanding commands, wait for them to complete */ 1186 while (!ocs_list_empty(&hw->cmd_head) && iters) { 1187 ocs_udelay(10000); 1188 ocs_hw_flush(hw); 1189 iters--; 1190 } 1191 1192 if (ocs_list_empty(&hw->cmd_head)) { 1193 ocs_log_debug(hw->os, "All commands completed on MQ queue\n"); 1194 } else { 1195 ocs_log_debug(hw->os, "Some commands still pending on MQ queue\n"); 1196 } 1197 1198 /* Cancel any remaining commands */ 1199 ocs_hw_command_cancel(hw); 1200 } else { 1201 hw->state = OCS_HW_STATE_TEARDOWN_IN_PROGRESS; 1202 } 1203 1204 ocs_lock_free(&hw->cmd_lock); 1205 1206 /* Free unregistered RPI if workaround is in force */ 1207 if (hw->workaround.use_unregistered_rpi) { 1208 sli_resource_free(&hw->sli, SLI_RSRC_FCOE_RPI, hw->workaround.unregistered_rid); 1209 } 1210 1211 max_rpi = sli_get_max_rsrc(&hw->sli, SLI_RSRC_FCOE_RPI); 1212 if (hw->rpi_ref) { 1213 for (i = 0; i < max_rpi; i++) { 1214 if (ocs_atomic_read(&hw->rpi_ref[i].rpi_count)) { 1215 ocs_log_debug(hw->os, "non-zero ref [%d]=%d\n", 1216 i, ocs_atomic_read(&hw->rpi_ref[i].rpi_count)); 1217 } 1218 } 1219 ocs_free(hw->os, hw->rpi_ref, max_rpi * sizeof(*hw->rpi_ref)); 1220 hw->rpi_ref = NULL; 1221 } 1222 1223 ocs_dma_free(hw->os, &hw->rnode_mem); 1224 1225 if (hw->io) { 1226 for (i = 0; i < hw->config.n_io; i++) { 1227 if (hw->io[i] && (hw->io[i]->sgl != NULL) && 1228 (hw->io[i]->sgl->virt != NULL)) { 1229 if(hw->io[i]->is_port_owned) { 1230 ocs_lock_free(&hw->io[i]->axr_lock); 1231 } 1232 ocs_dma_free(hw->os, hw->io[i]->sgl); 1233 } 1234 ocs_free(hw->os, hw->io[i], sizeof(ocs_hw_io_t)); 1235 hw->io[i] = NULL; 1236 } 1237 ocs_free(hw->os, hw->wqe_buffs, hw->config.n_io * hw->sli.config.wqe_size); 1238 hw->wqe_buffs = NULL; 1239 ocs_free(hw->os, hw->io, hw->config.n_io * sizeof(ocs_hw_io_t *)); 1240 hw->io = NULL; 1241 } 1242 1243 ocs_dma_free(hw->os, &hw->xfer_rdy); 1244 ocs_dma_free(hw->os, &hw->dump_sges); 1245 ocs_dma_free(hw->os, &hw->loop_map); 1246 1247 ocs_lock_free(&hw->io_lock); 1248 ocs_lock_free(&hw->io_abort_lock); 1249 1250 for (i = 0; i < hw->wq_count; i++) { 1251 sli_queue_free(&hw->sli, &hw->wq[i], destroy_queues, free_memory); 1252 } 1253 1254 for (i = 0; i < hw->rq_count; i++) { 1255 sli_queue_free(&hw->sli, &hw->rq[i], destroy_queues, free_memory); 1256 } 1257 1258 for (i = 0; i < hw->mq_count; i++) { 1259 sli_queue_free(&hw->sli, &hw->mq[i], destroy_queues, free_memory); 1260 } 1261 1262 for (i = 0; i < hw->cq_count; i++) { 1263 sli_queue_free(&hw->sli, &hw->cq[i], destroy_queues, free_memory); 1264 } 1265 1266 for (i = 0; i < hw->eq_count; i++) { 1267 sli_queue_free(&hw->sli, &hw->eq[i], destroy_queues, free_memory); 1268 } 1269 1270 ocs_hw_qtop_free(hw->qtop); 1271 1272 /* Free rq buffers */ 1273 ocs_hw_rx_free(hw); 1274 1275 hw_queue_teardown(hw); 1276 1277 ocs_hw_rqpair_teardown(hw); 1278 1279 if (sli_teardown(&hw->sli)) { 1280 ocs_log_err(hw->os, "SLI teardown failed\n"); 1281 } 1282 1283 ocs_queue_history_free(&hw->q_hist); 1284 1285 /* record the fact that the queues are non-functional */ 1286 hw->state = OCS_HW_STATE_UNINITIALIZED; 1287 1288 /* free sequence free pool */ 1289 ocs_array_free(hw->seq_pool); 1290 hw->seq_pool = NULL; 1291 1292 /* free hw_wq_callback pool */ 1293 ocs_pool_free(hw->wq_reqtag_pool); 1294 1295 ocs_dma_free(hw->os, &hw->domain_dmem); 1296 ocs_dma_free(hw->os, &hw->fcf_dmem); 1297 /* Mark HW setup as not having been called */ 1298 hw->hw_setup_called = FALSE; 1299 1300 return OCS_HW_RTN_SUCCESS; 1301 } 1302 1303 ocs_hw_rtn_e 1304 ocs_hw_reset(ocs_hw_t *hw, ocs_hw_reset_e reset) 1305 { 1306 uint32_t i; 1307 ocs_hw_rtn_e rc = OCS_HW_RTN_SUCCESS; 1308 uint32_t iters; 1309 ocs_hw_state_e prev_state = hw->state; 1310 1311 if (hw->state != OCS_HW_STATE_ACTIVE) { 1312 ocs_log_test(hw->os, "HW state %d is not active\n", hw->state); 1313 } 1314 1315 hw->state = OCS_HW_STATE_RESET_IN_PROGRESS; 1316 1317 /* shutdown target wqe timer */ 1318 shutdown_target_wqe_timer(hw); 1319 1320 ocs_hw_flush(hw); 1321 1322 /* 1323 * If an mailbox command requiring a DMA is outstanding (i.e. SFP/DDM), 1324 * then the FW will UE when the reset is issued. So attempt to complete 1325 * all mailbox commands. 1326 */ 1327 iters = 10; 1328 while (!ocs_list_empty(&hw->cmd_head) && iters) { 1329 ocs_udelay(10000); 1330 ocs_hw_flush(hw); 1331 iters--; 1332 } 1333 1334 if (ocs_list_empty(&hw->cmd_head)) { 1335 ocs_log_debug(hw->os, "All commands completed on MQ queue\n"); 1336 } else { 1337 ocs_log_debug(hw->os, "Some commands still pending on MQ queue\n"); 1338 } 1339 1340 /* Reset the chip */ 1341 switch(reset) { 1342 case OCS_HW_RESET_FUNCTION: 1343 ocs_log_debug(hw->os, "issuing function level reset\n"); 1344 if (sli_reset(&hw->sli)) { 1345 ocs_log_err(hw->os, "sli_reset failed\n"); 1346 rc = OCS_HW_RTN_ERROR; 1347 } 1348 break; 1349 case OCS_HW_RESET_FIRMWARE: 1350 ocs_log_debug(hw->os, "issuing firmware reset\n"); 1351 if (sli_fw_reset(&hw->sli)) { 1352 ocs_log_err(hw->os, "sli_soft_reset failed\n"); 1353 rc = OCS_HW_RTN_ERROR; 1354 } 1355 /* 1356 * Because the FW reset leaves the FW in a non-running state, 1357 * follow that with a regular reset. 1358 */ 1359 ocs_log_debug(hw->os, "issuing function level reset\n"); 1360 if (sli_reset(&hw->sli)) { 1361 ocs_log_err(hw->os, "sli_reset failed\n"); 1362 rc = OCS_HW_RTN_ERROR; 1363 } 1364 break; 1365 default: 1366 ocs_log_test(hw->os, "unknown reset type - no reset performed\n"); 1367 hw->state = prev_state; 1368 return OCS_HW_RTN_ERROR; 1369 } 1370 1371 /* Not safe to walk command/io lists unless they've been initialized */ 1372 if (prev_state != OCS_HW_STATE_UNINITIALIZED) { 1373 ocs_hw_command_cancel(hw); 1374 1375 /* Clean up the inuse list, the free list and the wait free list */ 1376 ocs_hw_io_cancel(hw); 1377 1378 ocs_memset(hw->domains, 0, sizeof(hw->domains)); 1379 ocs_memset(hw->fcf_index_fcfi, 0, sizeof(hw->fcf_index_fcfi)); 1380 1381 ocs_hw_link_event_init(hw); 1382 1383 ocs_lock(&hw->io_lock); 1384 /* The io lists should be empty, but remove any that didn't get cleaned up. */ 1385 while (!ocs_list_empty(&hw->io_timed_wqe)) { 1386 ocs_list_remove_head(&hw->io_timed_wqe); 1387 } 1388 /* Don't clean up the io_inuse list, the backend will do that when it finishes the IO */ 1389 1390 while (!ocs_list_empty(&hw->io_free)) { 1391 ocs_list_remove_head(&hw->io_free); 1392 } 1393 while (!ocs_list_empty(&hw->io_wait_free)) { 1394 ocs_list_remove_head(&hw->io_wait_free); 1395 } 1396 1397 /* Reset the request tag pool, the HW IO request tags are reassigned in ocs_hw_setup_io() */ 1398 ocs_hw_reqtag_reset(hw); 1399 1400 ocs_unlock(&hw->io_lock); 1401 } 1402 1403 if (prev_state != OCS_HW_STATE_UNINITIALIZED) { 1404 for (i = 0; i < hw->wq_count; i++) { 1405 sli_queue_reset(&hw->sli, &hw->wq[i]); 1406 } 1407 1408 for (i = 0; i < hw->rq_count; i++) { 1409 sli_queue_reset(&hw->sli, &hw->rq[i]); 1410 } 1411 1412 for (i = 0; i < hw->hw_rq_count; i++) { 1413 hw_rq_t *rq = hw->hw_rq[i]; 1414 if (rq->rq_tracker != NULL) { 1415 uint32_t j; 1416 1417 for (j = 0; j < rq->entry_count; j++) { 1418 rq->rq_tracker[j] = NULL; 1419 } 1420 } 1421 } 1422 1423 for (i = 0; i < hw->mq_count; i++) { 1424 sli_queue_reset(&hw->sli, &hw->mq[i]); 1425 } 1426 1427 for (i = 0; i < hw->cq_count; i++) { 1428 sli_queue_reset(&hw->sli, &hw->cq[i]); 1429 } 1430 1431 for (i = 0; i < hw->eq_count; i++) { 1432 sli_queue_reset(&hw->sli, &hw->eq[i]); 1433 } 1434 1435 /* Free rq buffers */ 1436 ocs_hw_rx_free(hw); 1437 1438 /* Teardown the HW queue topology */ 1439 hw_queue_teardown(hw); 1440 } else { 1441 /* Free rq buffers */ 1442 ocs_hw_rx_free(hw); 1443 } 1444 1445 /* 1446 * Re-apply the run-time workarounds after clearing the SLI config 1447 * fields in sli_reset. 1448 */ 1449 ocs_hw_workaround_setup(hw); 1450 hw->state = OCS_HW_STATE_QUEUES_ALLOCATED; 1451 1452 return rc; 1453 } 1454 1455 int32_t 1456 ocs_hw_get_num_eq(ocs_hw_t *hw) 1457 { 1458 return hw->eq_count; 1459 } 1460 1461 static int32_t 1462 ocs_hw_get_fw_timed_out(ocs_hw_t *hw) 1463 { 1464 /* The error values below are taken from LOWLEVEL_SET_WATCHDOG_TIMER_rev1.pdf 1465 * No further explanation is given in the document. 1466 * */ 1467 return (sli_reg_read(&hw->sli, SLI4_REG_SLIPORT_ERROR1) == 0x2 && 1468 sli_reg_read(&hw->sli, SLI4_REG_SLIPORT_ERROR2) == 0x10); 1469 } 1470 1471 ocs_hw_rtn_e 1472 ocs_hw_get(ocs_hw_t *hw, ocs_hw_property_e prop, uint32_t *value) 1473 { 1474 ocs_hw_rtn_e rc = OCS_HW_RTN_SUCCESS; 1475 int32_t tmp; 1476 1477 if (!value) { 1478 return OCS_HW_RTN_ERROR; 1479 } 1480 1481 *value = 0; 1482 1483 switch (prop) { 1484 case OCS_HW_N_IO: 1485 *value = hw->config.n_io; 1486 break; 1487 case OCS_HW_N_SGL: 1488 *value = (hw->config.n_sgl - SLI4_SGE_MAX_RESERVED); 1489 break; 1490 case OCS_HW_MAX_IO: 1491 *value = sli_get_max_rsrc(&hw->sli, SLI_RSRC_FCOE_XRI); 1492 break; 1493 case OCS_HW_MAX_NODES: 1494 *value = sli_get_max_rsrc(&hw->sli, SLI_RSRC_FCOE_RPI); 1495 break; 1496 case OCS_HW_MAX_RQ_ENTRIES: 1497 *value = hw->num_qentries[SLI_QTYPE_RQ]; 1498 break; 1499 case OCS_HW_RQ_DEFAULT_BUFFER_SIZE: 1500 *value = hw->config.rq_default_buffer_size; 1501 break; 1502 case OCS_HW_AUTO_XFER_RDY_CAPABLE: 1503 *value = sli_get_auto_xfer_rdy_capable(&hw->sli); 1504 break; 1505 case OCS_HW_AUTO_XFER_RDY_XRI_CNT: 1506 *value = hw->config.auto_xfer_rdy_xri_cnt; 1507 break; 1508 case OCS_HW_AUTO_XFER_RDY_SIZE: 1509 *value = hw->config.auto_xfer_rdy_size; 1510 break; 1511 case OCS_HW_AUTO_XFER_RDY_BLK_SIZE: 1512 switch (hw->config.auto_xfer_rdy_blk_size_chip) { 1513 case 0: 1514 *value = 512; 1515 break; 1516 case 1: 1517 *value = 1024; 1518 break; 1519 case 2: 1520 *value = 2048; 1521 break; 1522 case 3: 1523 *value = 4096; 1524 break; 1525 case 4: 1526 *value = 520; 1527 break; 1528 default: 1529 *value = 0; 1530 rc = OCS_HW_RTN_ERROR; 1531 break; 1532 } 1533 break; 1534 case OCS_HW_AUTO_XFER_RDY_T10_ENABLE: 1535 *value = hw->config.auto_xfer_rdy_t10_enable; 1536 break; 1537 case OCS_HW_AUTO_XFER_RDY_P_TYPE: 1538 *value = hw->config.auto_xfer_rdy_p_type; 1539 break; 1540 case OCS_HW_AUTO_XFER_RDY_REF_TAG_IS_LBA: 1541 *value = hw->config.auto_xfer_rdy_ref_tag_is_lba; 1542 break; 1543 case OCS_HW_AUTO_XFER_RDY_APP_TAG_VALID: 1544 *value = hw->config.auto_xfer_rdy_app_tag_valid; 1545 break; 1546 case OCS_HW_AUTO_XFER_RDY_APP_TAG_VALUE: 1547 *value = hw->config.auto_xfer_rdy_app_tag_value; 1548 break; 1549 case OCS_HW_MAX_SGE: 1550 *value = sli_get_max_sge(&hw->sli); 1551 break; 1552 case OCS_HW_MAX_SGL: 1553 *value = sli_get_max_sgl(&hw->sli); 1554 break; 1555 case OCS_HW_TOPOLOGY: 1556 /* 1557 * Infer link.status based on link.speed. 1558 * Report OCS_HW_TOPOLOGY_NONE if the link is down. 1559 */ 1560 if (hw->link.speed == 0) { 1561 *value = OCS_HW_TOPOLOGY_NONE; 1562 break; 1563 } 1564 switch (hw->link.topology) { 1565 case SLI_LINK_TOPO_NPORT: 1566 *value = OCS_HW_TOPOLOGY_NPORT; 1567 break; 1568 case SLI_LINK_TOPO_LOOP: 1569 *value = OCS_HW_TOPOLOGY_LOOP; 1570 break; 1571 case SLI_LINK_TOPO_NONE: 1572 *value = OCS_HW_TOPOLOGY_NONE; 1573 break; 1574 default: 1575 ocs_log_test(hw->os, "unsupported topology %#x\n", hw->link.topology); 1576 rc = OCS_HW_RTN_ERROR; 1577 break; 1578 } 1579 break; 1580 case OCS_HW_CONFIG_TOPOLOGY: 1581 *value = hw->config.topology; 1582 break; 1583 case OCS_HW_LINK_SPEED: 1584 *value = hw->link.speed; 1585 break; 1586 case OCS_HW_LINK_CONFIG_SPEED: 1587 switch (hw->config.speed) { 1588 case FC_LINK_SPEED_10G: 1589 *value = 10000; 1590 break; 1591 case FC_LINK_SPEED_AUTO_16_8_4: 1592 *value = 0; 1593 break; 1594 case FC_LINK_SPEED_2G: 1595 *value = 2000; 1596 break; 1597 case FC_LINK_SPEED_4G: 1598 *value = 4000; 1599 break; 1600 case FC_LINK_SPEED_8G: 1601 *value = 8000; 1602 break; 1603 case FC_LINK_SPEED_16G: 1604 *value = 16000; 1605 break; 1606 case FC_LINK_SPEED_32G: 1607 *value = 32000; 1608 break; 1609 default: 1610 ocs_log_test(hw->os, "unsupported speed %#x\n", hw->config.speed); 1611 rc = OCS_HW_RTN_ERROR; 1612 break; 1613 } 1614 break; 1615 case OCS_HW_IF_TYPE: 1616 *value = sli_get_if_type(&hw->sli); 1617 break; 1618 case OCS_HW_SLI_REV: 1619 *value = sli_get_sli_rev(&hw->sli); 1620 break; 1621 case OCS_HW_SLI_FAMILY: 1622 *value = sli_get_sli_family(&hw->sli); 1623 break; 1624 case OCS_HW_DIF_CAPABLE: 1625 *value = sli_get_dif_capable(&hw->sli); 1626 break; 1627 case OCS_HW_DIF_SEED: 1628 *value = hw->config.dif_seed; 1629 break; 1630 case OCS_HW_DIF_MODE: 1631 *value = hw->config.dif_mode; 1632 break; 1633 case OCS_HW_DIF_MULTI_SEPARATE: 1634 /* Lancer supports multiple DIF separates */ 1635 if (hw->sli.if_type == SLI4_IF_TYPE_LANCER_FC_ETH) { 1636 *value = TRUE; 1637 } else { 1638 *value = FALSE; 1639 } 1640 break; 1641 case OCS_HW_DUMP_MAX_SIZE: 1642 *value = hw->dump_size; 1643 break; 1644 case OCS_HW_DUMP_READY: 1645 *value = sli_dump_is_ready(&hw->sli); 1646 break; 1647 case OCS_HW_DUMP_PRESENT: 1648 *value = sli_dump_is_present(&hw->sli); 1649 break; 1650 case OCS_HW_RESET_REQUIRED: 1651 tmp = sli_reset_required(&hw->sli); 1652 if(tmp < 0) { 1653 rc = OCS_HW_RTN_ERROR; 1654 } else { 1655 *value = tmp; 1656 } 1657 break; 1658 case OCS_HW_FW_ERROR: 1659 *value = sli_fw_error_status(&hw->sli); 1660 break; 1661 case OCS_HW_FW_READY: 1662 *value = sli_fw_ready(&hw->sli); 1663 break; 1664 case OCS_HW_FW_TIMED_OUT: 1665 *value = ocs_hw_get_fw_timed_out(hw); 1666 break; 1667 case OCS_HW_HIGH_LOGIN_MODE: 1668 *value = sli_get_hlm_capable(&hw->sli); 1669 break; 1670 case OCS_HW_PREREGISTER_SGL: 1671 *value = sli_get_sgl_preregister_required(&hw->sli); 1672 break; 1673 case OCS_HW_HW_REV1: 1674 *value = sli_get_hw_revision(&hw->sli, 0); 1675 break; 1676 case OCS_HW_HW_REV2: 1677 *value = sli_get_hw_revision(&hw->sli, 1); 1678 break; 1679 case OCS_HW_HW_REV3: 1680 *value = sli_get_hw_revision(&hw->sli, 2); 1681 break; 1682 case OCS_HW_LINKCFG: 1683 *value = hw->linkcfg; 1684 break; 1685 case OCS_HW_ETH_LICENSE: 1686 *value = hw->eth_license; 1687 break; 1688 case OCS_HW_LINK_MODULE_TYPE: 1689 *value = sli_get_link_module_type(&hw->sli); 1690 break; 1691 case OCS_HW_NUM_CHUTES: 1692 *value = ocs_hw_get_num_chutes(hw); 1693 break; 1694 case OCS_HW_DISABLE_AR_TGT_DIF: 1695 *value = hw->workaround.disable_ar_tgt_dif; 1696 break; 1697 case OCS_HW_EMULATE_I_ONLY_AAB: 1698 *value = hw->config.i_only_aab; 1699 break; 1700 case OCS_HW_EMULATE_TARGET_WQE_TIMEOUT: 1701 *value = hw->config.emulate_tgt_wqe_timeout; 1702 break; 1703 case OCS_HW_VPD_LEN: 1704 *value = sli_get_vpd_len(&hw->sli); 1705 break; 1706 case OCS_HW_SGL_CHAINING_CAPABLE: 1707 *value = sli_get_is_sgl_chaining_capable(&hw->sli) || hw->workaround.sglc_misreported; 1708 break; 1709 case OCS_HW_SGL_CHAINING_ALLOWED: 1710 /* 1711 * SGL Chaining is allowed in the following cases: 1712 * 1. Lancer with host SGL Lists 1713 * 2. Skyhawk with pre-registered SGL Lists 1714 */ 1715 *value = FALSE; 1716 if ((sli_get_is_sgl_chaining_capable(&hw->sli) || hw->workaround.sglc_misreported) && 1717 !sli_get_sgl_preregister(&hw->sli) && 1718 SLI4_IF_TYPE_LANCER_FC_ETH == sli_get_if_type(&hw->sli)) { 1719 *value = TRUE; 1720 } 1721 1722 if ((sli_get_is_sgl_chaining_capable(&hw->sli) || hw->workaround.sglc_misreported) && 1723 sli_get_sgl_preregister(&hw->sli) && 1724 ((SLI4_IF_TYPE_BE3_SKH_PF == sli_get_if_type(&hw->sli)) || 1725 (SLI4_IF_TYPE_BE3_SKH_VF == sli_get_if_type(&hw->sli)))) { 1726 *value = TRUE; 1727 } 1728 break; 1729 case OCS_HW_SGL_CHAINING_HOST_ALLOCATED: 1730 /* Only lancer supports host allocated SGL Chaining buffers. */ 1731 *value = ((sli_get_is_sgl_chaining_capable(&hw->sli) || hw->workaround.sglc_misreported) && 1732 (SLI4_IF_TYPE_LANCER_FC_ETH == sli_get_if_type(&hw->sli))); 1733 break; 1734 case OCS_HW_SEND_FRAME_CAPABLE: 1735 if (hw->workaround.ignore_send_frame) { 1736 *value = 0; 1737 } else { 1738 /* Only lancer is capable */ 1739 *value = sli_get_if_type(&hw->sli) == SLI4_IF_TYPE_LANCER_FC_ETH; 1740 } 1741 break; 1742 case OCS_HW_RQ_SELECTION_POLICY: 1743 *value = hw->config.rq_selection_policy; 1744 break; 1745 case OCS_HW_RR_QUANTA: 1746 *value = hw->config.rr_quanta; 1747 break; 1748 case OCS_HW_MAX_VPORTS: 1749 *value = sli_get_max_rsrc(&hw->sli, SLI_RSRC_FCOE_VPI); 1750 break; 1751 default: 1752 ocs_log_test(hw->os, "unsupported property %#x\n", prop); 1753 rc = OCS_HW_RTN_ERROR; 1754 } 1755 1756 return rc; 1757 } 1758 1759 void * 1760 ocs_hw_get_ptr(ocs_hw_t *hw, ocs_hw_property_e prop) 1761 { 1762 void *rc = NULL; 1763 1764 switch (prop) { 1765 case OCS_HW_WWN_NODE: 1766 rc = sli_get_wwn_node(&hw->sli); 1767 break; 1768 case OCS_HW_WWN_PORT: 1769 rc = sli_get_wwn_port(&hw->sli); 1770 break; 1771 case OCS_HW_VPD: 1772 /* make sure VPD length is non-zero */ 1773 if (sli_get_vpd_len(&hw->sli)) { 1774 rc = sli_get_vpd(&hw->sli); 1775 } 1776 break; 1777 case OCS_HW_FW_REV: 1778 rc = sli_get_fw_name(&hw->sli, 0); 1779 break; 1780 case OCS_HW_FW_REV2: 1781 rc = sli_get_fw_name(&hw->sli, 1); 1782 break; 1783 case OCS_HW_IPL: 1784 rc = sli_get_ipl_name(&hw->sli); 1785 break; 1786 case OCS_HW_PORTNUM: 1787 rc = sli_get_portnum(&hw->sli); 1788 break; 1789 case OCS_HW_BIOS_VERSION_STRING: 1790 rc = sli_get_bios_version_string(&hw->sli); 1791 break; 1792 default: 1793 ocs_log_test(hw->os, "unsupported property %#x\n", prop); 1794 } 1795 1796 return rc; 1797 } 1798 1799 ocs_hw_rtn_e 1800 ocs_hw_set(ocs_hw_t *hw, ocs_hw_property_e prop, uint32_t value) 1801 { 1802 ocs_hw_rtn_e rc = OCS_HW_RTN_SUCCESS; 1803 1804 switch (prop) { 1805 case OCS_HW_N_IO: 1806 if (value > sli_get_max_rsrc(&hw->sli, SLI_RSRC_FCOE_XRI) || 1807 value == 0) { 1808 ocs_log_test(hw->os, "IO value out of range %d vs %d\n", 1809 value, sli_get_max_rsrc(&hw->sli, SLI_RSRC_FCOE_XRI)); 1810 rc = OCS_HW_RTN_ERROR; 1811 } else { 1812 hw->config.n_io = value; 1813 } 1814 break; 1815 case OCS_HW_N_SGL: 1816 value += SLI4_SGE_MAX_RESERVED; 1817 if (value > sli_get_max_sgl(&hw->sli)) { 1818 ocs_log_test(hw->os, "SGL value out of range %d vs %d\n", 1819 value, sli_get_max_sgl(&hw->sli)); 1820 rc = OCS_HW_RTN_ERROR; 1821 } else { 1822 hw->config.n_sgl = value; 1823 } 1824 break; 1825 case OCS_HW_TOPOLOGY: 1826 if ((sli_get_medium(&hw->sli) != SLI_LINK_MEDIUM_FC) && 1827 (value != OCS_HW_TOPOLOGY_AUTO)) { 1828 ocs_log_test(hw->os, "unsupported topology=%#x medium=%#x\n", 1829 value, sli_get_medium(&hw->sli)); 1830 rc = OCS_HW_RTN_ERROR; 1831 break; 1832 } 1833 1834 switch (value) { 1835 case OCS_HW_TOPOLOGY_AUTO: 1836 if (sli_get_medium(&hw->sli) == SLI_LINK_MEDIUM_FC) { 1837 sli_set_topology(&hw->sli, SLI4_READ_CFG_TOPO_FC); 1838 } else { 1839 sli_set_topology(&hw->sli, SLI4_READ_CFG_TOPO_FCOE); 1840 } 1841 break; 1842 case OCS_HW_TOPOLOGY_NPORT: 1843 sli_set_topology(&hw->sli, SLI4_READ_CFG_TOPO_FC_DA); 1844 break; 1845 case OCS_HW_TOPOLOGY_LOOP: 1846 sli_set_topology(&hw->sli, SLI4_READ_CFG_TOPO_FC_AL); 1847 break; 1848 default: 1849 ocs_log_test(hw->os, "unsupported topology %#x\n", value); 1850 rc = OCS_HW_RTN_ERROR; 1851 } 1852 hw->config.topology = value; 1853 break; 1854 case OCS_HW_LINK_SPEED: 1855 if (sli_get_medium(&hw->sli) != SLI_LINK_MEDIUM_FC) { 1856 switch (value) { 1857 case 0: /* Auto-speed negotiation */ 1858 case 10000: /* FCoE speed */ 1859 hw->config.speed = FC_LINK_SPEED_10G; 1860 break; 1861 default: 1862 ocs_log_test(hw->os, "unsupported speed=%#x medium=%#x\n", 1863 value, sli_get_medium(&hw->sli)); 1864 rc = OCS_HW_RTN_ERROR; 1865 } 1866 break; 1867 } 1868 1869 switch (value) { 1870 case 0: /* Auto-speed negotiation */ 1871 hw->config.speed = FC_LINK_SPEED_AUTO_16_8_4; 1872 break; 1873 case 2000: /* FC speeds */ 1874 hw->config.speed = FC_LINK_SPEED_2G; 1875 break; 1876 case 4000: 1877 hw->config.speed = FC_LINK_SPEED_4G; 1878 break; 1879 case 8000: 1880 hw->config.speed = FC_LINK_SPEED_8G; 1881 break; 1882 case 16000: 1883 hw->config.speed = FC_LINK_SPEED_16G; 1884 break; 1885 case 32000: 1886 hw->config.speed = FC_LINK_SPEED_32G; 1887 break; 1888 default: 1889 ocs_log_test(hw->os, "unsupported speed %d\n", value); 1890 rc = OCS_HW_RTN_ERROR; 1891 } 1892 break; 1893 case OCS_HW_DIF_SEED: 1894 /* Set the DIF seed - only for lancer right now */ 1895 if (SLI4_IF_TYPE_LANCER_FC_ETH != sli_get_if_type(&hw->sli)) { 1896 ocs_log_test(hw->os, "DIF seed not supported for this device\n"); 1897 rc = OCS_HW_RTN_ERROR; 1898 } else { 1899 hw->config.dif_seed = value; 1900 } 1901 break; 1902 case OCS_HW_DIF_MODE: 1903 switch (value) { 1904 case OCS_HW_DIF_MODE_INLINE: 1905 /* 1906 * Make sure we support inline DIF. 1907 * 1908 * Note: Having both bits clear means that we have old 1909 * FW that doesn't set the bits. 1910 */ 1911 if (sli_is_dif_inline_capable(&hw->sli)) { 1912 hw->config.dif_mode = value; 1913 } else { 1914 ocs_log_test(hw->os, "chip does not support DIF inline\n"); 1915 rc = OCS_HW_RTN_ERROR; 1916 } 1917 break; 1918 case OCS_HW_DIF_MODE_SEPARATE: 1919 /* Make sure we support DIF separates. */ 1920 if (sli_is_dif_separate_capable(&hw->sli)) { 1921 hw->config.dif_mode = value; 1922 } else { 1923 ocs_log_test(hw->os, "chip does not support DIF separate\n"); 1924 rc = OCS_HW_RTN_ERROR; 1925 } 1926 } 1927 break; 1928 case OCS_HW_RQ_PROCESS_LIMIT: { 1929 hw_rq_t *rq; 1930 uint32_t i; 1931 1932 /* For each hw_rq object, set its parent CQ limit value */ 1933 for (i = 0; i < hw->hw_rq_count; i++) { 1934 rq = hw->hw_rq[i]; 1935 hw->cq[rq->cq->instance].proc_limit = value; 1936 } 1937 break; 1938 } 1939 case OCS_HW_RQ_DEFAULT_BUFFER_SIZE: 1940 hw->config.rq_default_buffer_size = value; 1941 break; 1942 case OCS_HW_AUTO_XFER_RDY_XRI_CNT: 1943 hw->config.auto_xfer_rdy_xri_cnt = value; 1944 break; 1945 case OCS_HW_AUTO_XFER_RDY_SIZE: 1946 hw->config.auto_xfer_rdy_size = value; 1947 break; 1948 case OCS_HW_AUTO_XFER_RDY_BLK_SIZE: 1949 switch (value) { 1950 case 512: 1951 hw->config.auto_xfer_rdy_blk_size_chip = 0; 1952 break; 1953 case 1024: 1954 hw->config.auto_xfer_rdy_blk_size_chip = 1; 1955 break; 1956 case 2048: 1957 hw->config.auto_xfer_rdy_blk_size_chip = 2; 1958 break; 1959 case 4096: 1960 hw->config.auto_xfer_rdy_blk_size_chip = 3; 1961 break; 1962 case 520: 1963 hw->config.auto_xfer_rdy_blk_size_chip = 4; 1964 break; 1965 default: 1966 ocs_log_err(hw->os, "Invalid block size %d\n", 1967 value); 1968 rc = OCS_HW_RTN_ERROR; 1969 } 1970 break; 1971 case OCS_HW_AUTO_XFER_RDY_T10_ENABLE: 1972 hw->config.auto_xfer_rdy_t10_enable = value; 1973 break; 1974 case OCS_HW_AUTO_XFER_RDY_P_TYPE: 1975 hw->config.auto_xfer_rdy_p_type = value; 1976 break; 1977 case OCS_HW_AUTO_XFER_RDY_REF_TAG_IS_LBA: 1978 hw->config.auto_xfer_rdy_ref_tag_is_lba = value; 1979 break; 1980 case OCS_HW_AUTO_XFER_RDY_APP_TAG_VALID: 1981 hw->config.auto_xfer_rdy_app_tag_valid = value; 1982 break; 1983 case OCS_HW_AUTO_XFER_RDY_APP_TAG_VALUE: 1984 hw->config.auto_xfer_rdy_app_tag_value = value; 1985 break; 1986 case OCS_ESOC: 1987 hw->config.esoc = value; 1988 break; 1989 case OCS_HW_HIGH_LOGIN_MODE: 1990 rc = sli_set_hlm(&hw->sli, value); 1991 break; 1992 case OCS_HW_PREREGISTER_SGL: 1993 rc = sli_set_sgl_preregister(&hw->sli, value); 1994 break; 1995 case OCS_HW_ETH_LICENSE: 1996 hw->eth_license = value; 1997 break; 1998 case OCS_HW_EMULATE_I_ONLY_AAB: 1999 hw->config.i_only_aab = value; 2000 break; 2001 case OCS_HW_EMULATE_TARGET_WQE_TIMEOUT: 2002 hw->config.emulate_tgt_wqe_timeout = value; 2003 break; 2004 case OCS_HW_BOUNCE: 2005 hw->config.bounce = value; 2006 break; 2007 case OCS_HW_RQ_SELECTION_POLICY: 2008 hw->config.rq_selection_policy = value; 2009 break; 2010 case OCS_HW_RR_QUANTA: 2011 hw->config.rr_quanta = value; 2012 break; 2013 default: 2014 ocs_log_test(hw->os, "unsupported property %#x\n", prop); 2015 rc = OCS_HW_RTN_ERROR; 2016 } 2017 2018 return rc; 2019 } 2020 2021 ocs_hw_rtn_e 2022 ocs_hw_set_ptr(ocs_hw_t *hw, ocs_hw_property_e prop, void *value) 2023 { 2024 ocs_hw_rtn_e rc = OCS_HW_RTN_SUCCESS; 2025 2026 switch (prop) { 2027 case OCS_HW_WAR_VERSION: 2028 hw->hw_war_version = value; 2029 break; 2030 case OCS_HW_FILTER_DEF: { 2031 char *p = value; 2032 uint32_t idx = 0; 2033 2034 for (idx = 0; idx < ARRAY_SIZE(hw->config.filter_def); idx++) { 2035 hw->config.filter_def[idx] = 0; 2036 } 2037 2038 for (idx = 0; (idx < ARRAY_SIZE(hw->config.filter_def)) && (p != NULL) && *p; ) { 2039 hw->config.filter_def[idx++] = ocs_strtoul(p, 0, 0); 2040 p = ocs_strchr(p, ','); 2041 if (p != NULL) { 2042 p++; 2043 } 2044 } 2045 2046 break; 2047 } 2048 default: 2049 ocs_log_test(hw->os, "unsupported property %#x\n", prop); 2050 rc = OCS_HW_RTN_ERROR; 2051 break; 2052 } 2053 return rc; 2054 } 2055 /** 2056 * @ingroup interrupt 2057 * @brief Check for the events associated with the interrupt vector. 2058 * 2059 * @param hw Hardware context. 2060 * @param vector Zero-based interrupt vector number. 2061 * 2062 * @return Returns 0 on success, or a non-zero value on failure. 2063 */ 2064 int32_t 2065 ocs_hw_event_check(ocs_hw_t *hw, uint32_t vector) 2066 { 2067 int32_t rc = 0; 2068 2069 if (!hw) { 2070 ocs_log_err(NULL, "HW context NULL?!?\n"); 2071 return -1; 2072 } 2073 2074 if (vector > hw->eq_count) { 2075 ocs_log_err(hw->os, "vector %d. max %d\n", 2076 vector, hw->eq_count); 2077 return -1; 2078 } 2079 2080 /* 2081 * The caller should disable interrupts if they wish to prevent us 2082 * from processing during a shutdown. The following states are defined: 2083 * OCS_HW_STATE_UNINITIALIZED - No queues allocated 2084 * OCS_HW_STATE_QUEUES_ALLOCATED - The state after a chip reset, 2085 * queues are cleared. 2086 * OCS_HW_STATE_ACTIVE - Chip and queues are operational 2087 * OCS_HW_STATE_RESET_IN_PROGRESS - reset, we still want completions 2088 * OCS_HW_STATE_TEARDOWN_IN_PROGRESS - We still want mailbox 2089 * completions. 2090 */ 2091 if (hw->state != OCS_HW_STATE_UNINITIALIZED) { 2092 rc = sli_queue_is_empty(&hw->sli, &hw->eq[vector]); 2093 2094 /* Re-arm queue if there are no entries */ 2095 if (rc != 0) { 2096 sli_queue_arm(&hw->sli, &hw->eq[vector], TRUE); 2097 } 2098 } 2099 return rc; 2100 } 2101 2102 void 2103 ocs_hw_unsol_process_bounce(void *arg) 2104 { 2105 ocs_hw_sequence_t *seq = arg; 2106 ocs_hw_t *hw = seq->hw; 2107 2108 ocs_hw_assert(hw != NULL); 2109 ocs_hw_assert(hw->callback.unsolicited != NULL); 2110 2111 hw->callback.unsolicited(hw->args.unsolicited, seq); 2112 } 2113 2114 int32_t 2115 ocs_hw_process(ocs_hw_t *hw, uint32_t vector, uint32_t max_isr_time_msec) 2116 { 2117 hw_eq_t *eq; 2118 int32_t rc = 0; 2119 2120 CPUTRACE(""); 2121 2122 /* 2123 * The caller should disable interrupts if they wish to prevent us 2124 * from processing during a shutdown. The following states are defined: 2125 * OCS_HW_STATE_UNINITIALIZED - No queues allocated 2126 * OCS_HW_STATE_QUEUES_ALLOCATED - The state after a chip reset, 2127 * queues are cleared. 2128 * OCS_HW_STATE_ACTIVE - Chip and queues are operational 2129 * OCS_HW_STATE_RESET_IN_PROGRESS - reset, we still want completions 2130 * OCS_HW_STATE_TEARDOWN_IN_PROGRESS - We still want mailbox 2131 * completions. 2132 */ 2133 if (hw->state == OCS_HW_STATE_UNINITIALIZED) { 2134 return 0; 2135 } 2136 2137 /* Get pointer to hw_eq_t */ 2138 eq = hw->hw_eq[vector]; 2139 2140 OCS_STAT(eq->use_count++); 2141 2142 rc = ocs_hw_eq_process(hw, eq, max_isr_time_msec); 2143 2144 return rc; 2145 } 2146 2147 /** 2148 * @ingroup interrupt 2149 * @brief Process events associated with an EQ. 2150 * 2151 * @par Description 2152 * Loop termination: 2153 * @n @n Without a mechanism to terminate the completion processing loop, it 2154 * is possible under some workload conditions for the loop to never terminate 2155 * (or at least take longer than the OS is happy to have an interrupt handler 2156 * or kernel thread context hold a CPU without yielding). 2157 * @n @n The approach taken here is to periodically check how much time 2158 * we have been in this 2159 * processing loop, and if we exceed a predetermined time (multiple seconds), the 2160 * loop is terminated, and ocs_hw_process() returns. 2161 * 2162 * @param hw Hardware context. 2163 * @param eq Pointer to HW EQ object. 2164 * @param max_isr_time_msec Maximum time in msec to stay in this function. 2165 * 2166 * @return Returns 0 on success, or a non-zero value on failure. 2167 */ 2168 int32_t 2169 ocs_hw_eq_process(ocs_hw_t *hw, hw_eq_t *eq, uint32_t max_isr_time_msec) 2170 { 2171 uint8_t eqe[sizeof(sli4_eqe_t)] = { 0 }; 2172 uint32_t done = FALSE; 2173 uint32_t tcheck_count; 2174 time_t tstart; 2175 time_t telapsed; 2176 2177 tcheck_count = OCS_HW_TIMECHECK_ITERATIONS; 2178 tstart = ocs_msectime(); 2179 2180 CPUTRACE(""); 2181 2182 while (!done && !sli_queue_read(&hw->sli, eq->queue, eqe)) { 2183 uint16_t cq_id = 0; 2184 int32_t rc; 2185 2186 rc = sli_eq_parse(&hw->sli, eqe, &cq_id); 2187 if (unlikely(rc)) { 2188 if (rc > 0) { 2189 uint32_t i; 2190 2191 /* 2192 * Received a sentinel EQE indicating the EQ is full. 2193 * Process all CQs 2194 */ 2195 for (i = 0; i < hw->cq_count; i++) { 2196 ocs_hw_cq_process(hw, hw->hw_cq[i]); 2197 } 2198 continue; 2199 } else { 2200 return rc; 2201 } 2202 } else { 2203 int32_t index = ocs_hw_queue_hash_find(hw->cq_hash, cq_id); 2204 if (likely(index >= 0)) { 2205 ocs_hw_cq_process(hw, hw->hw_cq[index]); 2206 } else { 2207 ocs_log_err(hw->os, "bad CQ_ID %#06x\n", cq_id); 2208 } 2209 } 2210 2211 if (eq->queue->n_posted > (eq->queue->posted_limit)) { 2212 sli_queue_arm(&hw->sli, eq->queue, FALSE); 2213 } 2214 2215 if (tcheck_count && (--tcheck_count == 0)) { 2216 tcheck_count = OCS_HW_TIMECHECK_ITERATIONS; 2217 telapsed = ocs_msectime() - tstart; 2218 if (telapsed >= max_isr_time_msec) { 2219 done = TRUE; 2220 } 2221 } 2222 } 2223 sli_queue_eq_arm(&hw->sli, eq->queue, TRUE); 2224 2225 return 0; 2226 } 2227 2228 /** 2229 * @brief Submit queued (pending) mbx commands. 2230 * 2231 * @par Description 2232 * Submit queued mailbox commands. 2233 * --- Assumes that hw->cmd_lock is held --- 2234 * 2235 * @param hw Hardware context. 2236 * 2237 * @return Returns 0 on success, or a negative error code value on failure. 2238 */ 2239 static int32_t 2240 ocs_hw_cmd_submit_pending(ocs_hw_t *hw) 2241 { 2242 ocs_command_ctx_t *ctx; 2243 int32_t rc = 0; 2244 2245 /* Assumes lock held */ 2246 2247 /* Only submit MQE if there's room */ 2248 while (hw->cmd_head_count < (OCS_HW_MQ_DEPTH - 1)) { 2249 ctx = ocs_list_remove_head(&hw->cmd_pending); 2250 if (ctx == NULL) { 2251 break; 2252 } 2253 ocs_list_add_tail(&hw->cmd_head, ctx); 2254 hw->cmd_head_count++; 2255 if (sli_queue_write(&hw->sli, hw->mq, ctx->buf) < 0) { 2256 ocs_log_test(hw->os, "sli_queue_write failed: %d\n", rc); 2257 rc = -1; 2258 break; 2259 } 2260 } 2261 return rc; 2262 } 2263 2264 /** 2265 * @ingroup io 2266 * @brief Issue a SLI command. 2267 * 2268 * @par Description 2269 * Send a mailbox command to the hardware, and either wait for a completion 2270 * (OCS_CMD_POLL) or get an optional asynchronous completion (OCS_CMD_NOWAIT). 2271 * 2272 * @param hw Hardware context. 2273 * @param cmd Buffer containing a formatted command and results. 2274 * @param opts Command options: 2275 * - OCS_CMD_POLL - Command executes synchronously and busy-waits for the completion. 2276 * - OCS_CMD_NOWAIT - Command executes asynchronously. Uses callback. 2277 * @param cb Function callback used for asynchronous mode. May be NULL. 2278 * @n Prototype is <tt>(*cb)(void *arg, uint8_t *cmd)</tt>. 2279 * @n @n @b Note: If the 2280 * callback function pointer is NULL, the results of the command are silently 2281 * discarded, allowing this pointer to exist solely on the stack. 2282 * @param arg Argument passed to an asynchronous callback. 2283 * 2284 * @return Returns 0 on success, or a non-zero value on failure. 2285 */ 2286 ocs_hw_rtn_e 2287 ocs_hw_command(ocs_hw_t *hw, uint8_t *cmd, uint32_t opts, void *cb, void *arg) 2288 { 2289 ocs_hw_rtn_e rc = OCS_HW_RTN_ERROR; 2290 2291 /* 2292 * If the chip is in an error state (UE'd) then reject this mailbox 2293 * command. 2294 */ 2295 if (sli_fw_error_status(&hw->sli) > 0) { 2296 uint32_t err1 = sli_reg_read(&hw->sli, SLI4_REG_SLIPORT_ERROR1); 2297 uint32_t err2 = sli_reg_read(&hw->sli, SLI4_REG_SLIPORT_ERROR2); 2298 if (hw->expiration_logged == 0 && err1 == 0x2 && err2 == 0x10) { 2299 hw->expiration_logged = 1; 2300 ocs_log_crit(hw->os,"Emulex: Heartbeat expired after %d seconds\n", 2301 hw->watchdog_timeout); 2302 } 2303 ocs_log_crit(hw->os, "Chip is in an error state - reset needed\n"); 2304 ocs_log_crit(hw->os, "status=%#x error1=%#x error2=%#x\n", 2305 sli_reg_read(&hw->sli, SLI4_REG_SLIPORT_STATUS), 2306 err1, err2); 2307 2308 return OCS_HW_RTN_ERROR; 2309 } 2310 2311 if (OCS_CMD_POLL == opts) { 2312 ocs_lock(&hw->cmd_lock); 2313 if (hw->mq->length && !sli_queue_is_empty(&hw->sli, hw->mq)) { 2314 /* 2315 * Can't issue Boot-strap mailbox command with other 2316 * mail-queue commands pending as this interaction is 2317 * undefined 2318 */ 2319 rc = OCS_HW_RTN_ERROR; 2320 } else { 2321 void *bmbx = hw->sli.bmbx.virt; 2322 2323 ocs_memset(bmbx, 0, SLI4_BMBX_SIZE); 2324 ocs_memcpy(bmbx, cmd, SLI4_BMBX_SIZE); 2325 2326 if (sli_bmbx_command(&hw->sli) == 0) { 2327 rc = OCS_HW_RTN_SUCCESS; 2328 ocs_memcpy(cmd, bmbx, SLI4_BMBX_SIZE); 2329 } 2330 } 2331 ocs_unlock(&hw->cmd_lock); 2332 } else if (OCS_CMD_NOWAIT == opts) { 2333 ocs_command_ctx_t *ctx = NULL; 2334 2335 ctx = ocs_malloc(hw->os, sizeof(ocs_command_ctx_t), OCS_M_ZERO | OCS_M_NOWAIT); 2336 if (!ctx) { 2337 ocs_log_err(hw->os, "can't allocate command context\n"); 2338 return OCS_HW_RTN_NO_RESOURCES; 2339 } 2340 2341 if (hw->state != OCS_HW_STATE_ACTIVE) { 2342 ocs_log_err(hw->os, "Can't send command, HW state=%d\n", hw->state); 2343 ocs_free(hw->os, ctx, sizeof(*ctx)); 2344 return OCS_HW_RTN_ERROR; 2345 } 2346 2347 if (cb) { 2348 ctx->cb = cb; 2349 ctx->arg = arg; 2350 } 2351 ctx->buf = cmd; 2352 ctx->ctx = hw; 2353 2354 ocs_lock(&hw->cmd_lock); 2355 2356 /* Add to pending list */ 2357 ocs_list_add_tail(&hw->cmd_pending, ctx); 2358 2359 /* Submit as much of the pending list as we can */ 2360 if (ocs_hw_cmd_submit_pending(hw) == 0) { 2361 rc = OCS_HW_RTN_SUCCESS; 2362 } 2363 2364 ocs_unlock(&hw->cmd_lock); 2365 } 2366 2367 return rc; 2368 } 2369 2370 /** 2371 * @ingroup devInitShutdown 2372 * @brief Register a callback for the given event. 2373 * 2374 * @param hw Hardware context. 2375 * @param which Event of interest. 2376 * @param func Function to call when the event occurs. 2377 * @param arg Argument passed to the callback function. 2378 * 2379 * @return Returns 0 on success, or a non-zero value on failure. 2380 */ 2381 ocs_hw_rtn_e 2382 ocs_hw_callback(ocs_hw_t *hw, ocs_hw_callback_e which, void *func, void *arg) 2383 { 2384 2385 if (!hw || !func || (which >= OCS_HW_CB_MAX)) { 2386 ocs_log_err(NULL, "bad parameter hw=%p which=%#x func=%p\n", 2387 hw, which, func); 2388 return OCS_HW_RTN_ERROR; 2389 } 2390 2391 switch (which) { 2392 case OCS_HW_CB_DOMAIN: 2393 hw->callback.domain = func; 2394 hw->args.domain = arg; 2395 break; 2396 case OCS_HW_CB_PORT: 2397 hw->callback.port = func; 2398 hw->args.port = arg; 2399 break; 2400 case OCS_HW_CB_UNSOLICITED: 2401 hw->callback.unsolicited = func; 2402 hw->args.unsolicited = arg; 2403 break; 2404 case OCS_HW_CB_REMOTE_NODE: 2405 hw->callback.rnode = func; 2406 hw->args.rnode = arg; 2407 break; 2408 case OCS_HW_CB_BOUNCE: 2409 hw->callback.bounce = func; 2410 hw->args.bounce = arg; 2411 break; 2412 default: 2413 ocs_log_test(hw->os, "unknown callback %#x\n", which); 2414 return OCS_HW_RTN_ERROR; 2415 } 2416 2417 return OCS_HW_RTN_SUCCESS; 2418 } 2419 2420 /** 2421 * @ingroup port 2422 * @brief Allocate a port object. 2423 * 2424 * @par Description 2425 * This function allocates a VPI object for the port and stores it in the 2426 * indicator field of the port object. 2427 * 2428 * @param hw Hardware context. 2429 * @param sport SLI port object used to connect to the domain. 2430 * @param domain Domain object associated with this port (may be NULL). 2431 * @param wwpn Port's WWPN in big-endian order, or NULL to use default. 2432 * 2433 * @return Returns 0 on success, or a non-zero value on failure. 2434 */ 2435 ocs_hw_rtn_e 2436 ocs_hw_port_alloc(ocs_hw_t *hw, ocs_sli_port_t *sport, ocs_domain_t *domain, 2437 uint8_t *wwpn) 2438 { 2439 uint8_t *cmd = NULL; 2440 ocs_hw_rtn_e rc = OCS_HW_RTN_SUCCESS; 2441 uint32_t index; 2442 2443 sport->indicator = UINT32_MAX; 2444 sport->hw = hw; 2445 sport->ctx.app = sport; 2446 sport->sm_free_req_pending = 0; 2447 2448 /* 2449 * Check if the chip is in an error state (UE'd) before proceeding. 2450 */ 2451 if (sli_fw_error_status(&hw->sli) > 0) { 2452 ocs_log_crit(hw->os, "Chip is in an error state - reset needed\n"); 2453 return OCS_HW_RTN_ERROR; 2454 } 2455 2456 if (wwpn) { 2457 ocs_memcpy(&sport->sli_wwpn, wwpn, sizeof(sport->sli_wwpn)); 2458 } 2459 2460 if (sli_resource_alloc(&hw->sli, SLI_RSRC_FCOE_VPI, &sport->indicator, &index)) { 2461 ocs_log_err(hw->os, "FCOE_VPI allocation failure\n"); 2462 return OCS_HW_RTN_ERROR; 2463 } 2464 2465 if (domain != NULL) { 2466 ocs_sm_function_t next = NULL; 2467 2468 cmd = ocs_malloc(hw->os, SLI4_BMBX_SIZE, OCS_M_ZERO | OCS_M_NOWAIT); 2469 if (!cmd) { 2470 ocs_log_err(hw->os, "command memory allocation failed\n"); 2471 rc = OCS_HW_RTN_NO_MEMORY; 2472 goto ocs_hw_port_alloc_out; 2473 } 2474 2475 /* If the WWPN is NULL, fetch the default WWPN and WWNN before 2476 * initializing the VPI 2477 */ 2478 if (!wwpn) { 2479 next = __ocs_hw_port_alloc_read_sparm64; 2480 } else { 2481 next = __ocs_hw_port_alloc_init_vpi; 2482 } 2483 2484 ocs_sm_transition(&sport->ctx, next, cmd); 2485 } else if (!wwpn) { 2486 /* This is the convention for the HW, not SLI */ 2487 ocs_log_test(hw->os, "need WWN for physical port\n"); 2488 rc = OCS_HW_RTN_ERROR; 2489 } else { 2490 /* domain NULL and wwpn non-NULL */ 2491 ocs_sm_transition(&sport->ctx, __ocs_hw_port_alloc_init, NULL); 2492 } 2493 2494 ocs_hw_port_alloc_out: 2495 if (rc != OCS_HW_RTN_SUCCESS) { 2496 ocs_free(hw->os, cmd, SLI4_BMBX_SIZE); 2497 2498 sli_resource_free(&hw->sli, SLI_RSRC_FCOE_VPI, sport->indicator); 2499 } 2500 2501 return rc; 2502 } 2503 2504 /** 2505 * @ingroup port 2506 * @brief Attach a physical/virtual SLI port to a domain. 2507 * 2508 * @par Description 2509 * This function registers a previously-allocated VPI with the 2510 * device. 2511 * 2512 * @param hw Hardware context. 2513 * @param sport Pointer to the SLI port object. 2514 * @param fc_id Fibre Channel ID to associate with this port. 2515 * 2516 * @return Returns OCS_HW_RTN_SUCCESS on success, or an error code on failure. 2517 */ 2518 ocs_hw_rtn_e 2519 ocs_hw_port_attach(ocs_hw_t *hw, ocs_sli_port_t *sport, uint32_t fc_id) 2520 { 2521 uint8_t *buf = NULL; 2522 ocs_hw_rtn_e rc = OCS_HW_RTN_SUCCESS; 2523 2524 if (!hw || !sport) { 2525 ocs_log_err(hw ? hw->os : NULL, 2526 "bad parameter(s) hw=%p sport=%p\n", hw, 2527 sport); 2528 return OCS_HW_RTN_ERROR; 2529 } 2530 2531 /* 2532 * Check if the chip is in an error state (UE'd) before proceeding. 2533 */ 2534 if (sli_fw_error_status(&hw->sli) > 0) { 2535 ocs_log_crit(hw->os, "Chip is in an error state - reset needed\n"); 2536 return OCS_HW_RTN_ERROR; 2537 } 2538 2539 buf = ocs_malloc(hw->os, SLI4_BMBX_SIZE, OCS_M_NOWAIT); 2540 if (!buf) { 2541 ocs_log_err(hw->os, "no buffer for command\n"); 2542 return OCS_HW_RTN_NO_MEMORY; 2543 } 2544 2545 sport->fc_id = fc_id; 2546 ocs_sm_post_event(&sport->ctx, OCS_EVT_HW_PORT_REQ_ATTACH, buf); 2547 return rc; 2548 } 2549 2550 /** 2551 * @brief Called when the port control command completes. 2552 * 2553 * @par Description 2554 * We only need to free the mailbox command buffer. 2555 * 2556 * @param hw Hardware context. 2557 * @param status Status field from the mbox completion. 2558 * @param mqe Mailbox response structure. 2559 * @param arg Pointer to a callback function that signals the caller that the command is done. 2560 * 2561 * @return Returns 0. 2562 */ 2563 static int32_t 2564 ocs_hw_cb_port_control(ocs_hw_t *hw, int32_t status, uint8_t *mqe, void *arg) 2565 { 2566 ocs_free(hw->os, mqe, SLI4_BMBX_SIZE); 2567 return 0; 2568 } 2569 2570 /** 2571 * @ingroup port 2572 * @brief Control a port (initialize, shutdown, or set link configuration). 2573 * 2574 * @par Description 2575 * This function controls a port depending on the @c ctrl parameter: 2576 * - @b OCS_HW_PORT_INIT - 2577 * Issues the CONFIG_LINK and INIT_LINK commands for the specified port. 2578 * The HW generates an OCS_HW_DOMAIN_FOUND event when the link comes up. 2579 * . 2580 * - @b OCS_HW_PORT_SHUTDOWN - 2581 * Issues the DOWN_LINK command for the specified port. 2582 * The HW generates an OCS_HW_DOMAIN_LOST event when the link is down. 2583 * . 2584 * - @b OCS_HW_PORT_SET_LINK_CONFIG - 2585 * Sets the link configuration. 2586 * 2587 * @param hw Hardware context. 2588 * @param ctrl Specifies the operation: 2589 * - OCS_HW_PORT_INIT 2590 * - OCS_HW_PORT_SHUTDOWN 2591 * - OCS_HW_PORT_SET_LINK_CONFIG 2592 * 2593 * @param value Operation-specific value. 2594 * - OCS_HW_PORT_INIT - Selective reset AL_PA 2595 * - OCS_HW_PORT_SHUTDOWN - N/A 2596 * - OCS_HW_PORT_SET_LINK_CONFIG - An enum #ocs_hw_linkcfg_e value. 2597 * 2598 * @param cb Callback function to invoke the following operation. 2599 * - OCS_HW_PORT_INIT/OCS_HW_PORT_SHUTDOWN - NULL (link events 2600 * are handled by the OCS_HW_CB_DOMAIN callbacks). 2601 * - OCS_HW_PORT_SET_LINK_CONFIG - Invoked after linkcfg mailbox command 2602 * completes. 2603 * 2604 * @param arg Callback argument invoked after the command completes. 2605 * - OCS_HW_PORT_INIT/OCS_HW_PORT_SHUTDOWN - NULL (link events 2606 * are handled by the OCS_HW_CB_DOMAIN callbacks). 2607 * - OCS_HW_PORT_SET_LINK_CONFIG - Invoked after linkcfg mailbox command 2608 * completes. 2609 * 2610 * @return Returns 0 on success, or a non-zero value on failure. 2611 */ 2612 ocs_hw_rtn_e 2613 ocs_hw_port_control(ocs_hw_t *hw, ocs_hw_port_e ctrl, uintptr_t value, ocs_hw_port_control_cb_t cb, void *arg) 2614 { 2615 ocs_hw_rtn_e rc = OCS_HW_RTN_ERROR; 2616 2617 switch (ctrl) { 2618 case OCS_HW_PORT_INIT: 2619 { 2620 uint8_t *init_link; 2621 uint32_t speed = 0; 2622 uint8_t reset_alpa = 0; 2623 2624 if (SLI_LINK_MEDIUM_FC == sli_get_medium(&hw->sli)) { 2625 uint8_t *cfg_link; 2626 2627 cfg_link = ocs_malloc(hw->os, SLI4_BMBX_SIZE, OCS_M_NOWAIT); 2628 if (cfg_link == NULL) { 2629 ocs_log_err(hw->os, "no buffer for command\n"); 2630 return OCS_HW_RTN_NO_MEMORY; 2631 } 2632 2633 if (sli_cmd_config_link(&hw->sli, cfg_link, SLI4_BMBX_SIZE)) { 2634 rc = ocs_hw_command(hw, cfg_link, OCS_CMD_NOWAIT, 2635 ocs_hw_cb_port_control, NULL); 2636 } 2637 2638 if (rc != OCS_HW_RTN_SUCCESS) { 2639 ocs_free(hw->os, cfg_link, SLI4_BMBX_SIZE); 2640 ocs_log_err(hw->os, "CONFIG_LINK failed\n"); 2641 break; 2642 } 2643 speed = hw->config.speed; 2644 reset_alpa = (uint8_t)(value & 0xff); 2645 } else { 2646 speed = FC_LINK_SPEED_10G; 2647 } 2648 2649 /* 2650 * Bring link up, unless FW version is not supported 2651 */ 2652 if (hw->workaround.fw_version_too_low) { 2653 if (SLI4_IF_TYPE_LANCER_FC_ETH == hw->sli.if_type) { 2654 ocs_log_err(hw->os, "Cannot bring up link. Please update firmware to %s or later (current version is %s)\n", 2655 OCS_FW_VER_STR(OCS_MIN_FW_VER_LANCER), (char *) sli_get_fw_name(&hw->sli,0)); 2656 } else { 2657 ocs_log_err(hw->os, "Cannot bring up link. Please update firmware to %s or later (current version is %s)\n", 2658 OCS_FW_VER_STR(OCS_MIN_FW_VER_SKYHAWK), (char *) sli_get_fw_name(&hw->sli, 0)); 2659 } 2660 2661 return OCS_HW_RTN_ERROR; 2662 } 2663 2664 rc = OCS_HW_RTN_ERROR; 2665 2666 /* Allocate a new buffer for the init_link command */ 2667 init_link = ocs_malloc(hw->os, SLI4_BMBX_SIZE, OCS_M_NOWAIT); 2668 if (init_link == NULL) { 2669 ocs_log_err(hw->os, "no buffer for command\n"); 2670 return OCS_HW_RTN_NO_MEMORY; 2671 } 2672 2673 if (sli_cmd_init_link(&hw->sli, init_link, SLI4_BMBX_SIZE, speed, reset_alpa)) { 2674 rc = ocs_hw_command(hw, init_link, OCS_CMD_NOWAIT, 2675 ocs_hw_cb_port_control, NULL); 2676 } 2677 /* Free buffer on error, since no callback is coming */ 2678 if (rc != OCS_HW_RTN_SUCCESS) { 2679 ocs_free(hw->os, init_link, SLI4_BMBX_SIZE); 2680 ocs_log_err(hw->os, "INIT_LINK failed\n"); 2681 } 2682 break; 2683 } 2684 case OCS_HW_PORT_SHUTDOWN: 2685 { 2686 uint8_t *down_link; 2687 2688 down_link = ocs_malloc(hw->os, SLI4_BMBX_SIZE, OCS_M_NOWAIT); 2689 if (down_link == NULL) { 2690 ocs_log_err(hw->os, "no buffer for command\n"); 2691 return OCS_HW_RTN_NO_MEMORY; 2692 } 2693 if (sli_cmd_down_link(&hw->sli, down_link, SLI4_BMBX_SIZE)) { 2694 rc = ocs_hw_command(hw, down_link, OCS_CMD_NOWAIT, 2695 ocs_hw_cb_port_control, NULL); 2696 } 2697 /* Free buffer on error, since no callback is coming */ 2698 if (rc != OCS_HW_RTN_SUCCESS) { 2699 ocs_free(hw->os, down_link, SLI4_BMBX_SIZE); 2700 ocs_log_err(hw->os, "DOWN_LINK failed\n"); 2701 } 2702 break; 2703 } 2704 case OCS_HW_PORT_SET_LINK_CONFIG: 2705 rc = ocs_hw_set_linkcfg(hw, (ocs_hw_linkcfg_e)value, OCS_CMD_NOWAIT, cb, arg); 2706 break; 2707 default: 2708 ocs_log_test(hw->os, "unhandled control %#x\n", ctrl); 2709 break; 2710 } 2711 2712 return rc; 2713 } 2714 2715 /** 2716 * @ingroup port 2717 * @brief Free port resources. 2718 * 2719 * @par Description 2720 * Issue the UNREG_VPI command to free the assigned VPI context. 2721 * 2722 * @param hw Hardware context. 2723 * @param sport SLI port object used to connect to the domain. 2724 * 2725 * @return Returns 0 on success, or a non-zero value on failure. 2726 */ 2727 ocs_hw_rtn_e 2728 ocs_hw_port_free(ocs_hw_t *hw, ocs_sli_port_t *sport) 2729 { 2730 ocs_hw_rtn_e rc = OCS_HW_RTN_SUCCESS; 2731 2732 if (!hw || !sport) { 2733 ocs_log_err(hw ? hw->os : NULL, 2734 "bad parameter(s) hw=%p sport=%p\n", hw, 2735 sport); 2736 return OCS_HW_RTN_ERROR; 2737 } 2738 2739 /* 2740 * Check if the chip is in an error state (UE'd) before proceeding. 2741 */ 2742 if (sli_fw_error_status(&hw->sli) > 0) { 2743 ocs_log_crit(hw->os, "Chip is in an error state - reset needed\n"); 2744 return OCS_HW_RTN_ERROR; 2745 } 2746 2747 ocs_sm_post_event(&sport->ctx, OCS_EVT_HW_PORT_REQ_FREE, NULL); 2748 return rc; 2749 } 2750 2751 /** 2752 * @ingroup domain 2753 * @brief Allocate a fabric domain object. 2754 * 2755 * @par Description 2756 * This function starts a series of commands needed to connect to the domain, including 2757 * - REG_FCFI 2758 * - INIT_VFI 2759 * - READ_SPARMS 2760 * . 2761 * @b Note: Not all SLI interface types use all of the above commands. 2762 * @n @n Upon successful allocation, the HW generates a OCS_HW_DOMAIN_ALLOC_OK 2763 * event. On failure, it generates a OCS_HW_DOMAIN_ALLOC_FAIL event. 2764 * 2765 * @param hw Hardware context. 2766 * @param domain Pointer to the domain object. 2767 * @param fcf FCF index. 2768 * @param vlan VLAN ID. 2769 * 2770 * @return Returns 0 on success, or a non-zero value on failure. 2771 */ 2772 ocs_hw_rtn_e 2773 ocs_hw_domain_alloc(ocs_hw_t *hw, ocs_domain_t *domain, uint32_t fcf, uint32_t vlan) 2774 { 2775 uint8_t *cmd = NULL; 2776 uint32_t index; 2777 2778 if (!hw || !domain || !domain->sport) { 2779 ocs_log_err(NULL, "bad parameter(s) hw=%p domain=%p sport=%p\n", 2780 hw, domain, domain ? domain->sport : NULL); 2781 return OCS_HW_RTN_ERROR; 2782 } 2783 2784 /* 2785 * Check if the chip is in an error state (UE'd) before proceeding. 2786 */ 2787 if (sli_fw_error_status(&hw->sli) > 0) { 2788 ocs_log_crit(hw->os, "Chip is in an error state - reset needed\n"); 2789 return OCS_HW_RTN_ERROR; 2790 } 2791 2792 cmd = ocs_malloc(hw->os, SLI4_BMBX_SIZE, OCS_M_ZERO | OCS_M_NOWAIT); 2793 if (!cmd) { 2794 ocs_log_err(hw->os, "command memory allocation failed\n"); 2795 return OCS_HW_RTN_NO_MEMORY; 2796 } 2797 2798 domain->dma = hw->domain_dmem; 2799 2800 domain->hw = hw; 2801 domain->sm.app = domain; 2802 domain->fcf = fcf; 2803 domain->fcf_indicator = UINT32_MAX; 2804 domain->vlan_id = vlan; 2805 domain->indicator = UINT32_MAX; 2806 2807 if (sli_resource_alloc(&hw->sli, SLI_RSRC_FCOE_VFI, &domain->indicator, &index)) { 2808 ocs_log_err(hw->os, "FCOE_VFI allocation failure\n"); 2809 2810 ocs_free(hw->os, cmd, SLI4_BMBX_SIZE); 2811 2812 return OCS_HW_RTN_ERROR; 2813 } 2814 2815 ocs_sm_transition(&domain->sm, __ocs_hw_domain_init, cmd); 2816 return OCS_HW_RTN_SUCCESS; 2817 } 2818 2819 /** 2820 * @ingroup domain 2821 * @brief Attach a SLI port to a domain. 2822 * 2823 * @param hw Hardware context. 2824 * @param domain Pointer to the domain object. 2825 * @param fc_id Fibre Channel ID to associate with this port. 2826 * 2827 * @return Returns 0 on success, or a non-zero value on failure. 2828 */ 2829 ocs_hw_rtn_e 2830 ocs_hw_domain_attach(ocs_hw_t *hw, ocs_domain_t *domain, uint32_t fc_id) 2831 { 2832 uint8_t *buf = NULL; 2833 ocs_hw_rtn_e rc = OCS_HW_RTN_SUCCESS; 2834 2835 if (!hw || !domain) { 2836 ocs_log_err(hw ? hw->os : NULL, 2837 "bad parameter(s) hw=%p domain=%p\n", 2838 hw, domain); 2839 return OCS_HW_RTN_ERROR; 2840 } 2841 2842 /* 2843 * Check if the chip is in an error state (UE'd) before proceeding. 2844 */ 2845 if (sli_fw_error_status(&hw->sli) > 0) { 2846 ocs_log_crit(hw->os, "Chip is in an error state - reset needed\n"); 2847 return OCS_HW_RTN_ERROR; 2848 } 2849 2850 buf = ocs_malloc(hw->os, SLI4_BMBX_SIZE, OCS_M_NOWAIT); 2851 if (!buf) { 2852 ocs_log_err(hw->os, "no buffer for command\n"); 2853 return OCS_HW_RTN_NO_MEMORY; 2854 } 2855 2856 domain->sport->fc_id = fc_id; 2857 ocs_sm_post_event(&domain->sm, OCS_EVT_HW_DOMAIN_REQ_ATTACH, buf); 2858 return rc; 2859 } 2860 2861 /** 2862 * @ingroup domain 2863 * @brief Free a fabric domain object. 2864 * 2865 * @par Description 2866 * Free both the driver and SLI port resources associated with the domain. 2867 * 2868 * @param hw Hardware context. 2869 * @param domain Pointer to the domain object. 2870 * 2871 * @return Returns 0 on success, or a non-zero value on failure. 2872 */ 2873 ocs_hw_rtn_e 2874 ocs_hw_domain_free(ocs_hw_t *hw, ocs_domain_t *domain) 2875 { 2876 ocs_hw_rtn_e rc = OCS_HW_RTN_SUCCESS; 2877 2878 if (!hw || !domain) { 2879 ocs_log_err(hw ? hw->os : NULL, 2880 "bad parameter(s) hw=%p domain=%p\n", 2881 hw, domain); 2882 return OCS_HW_RTN_ERROR; 2883 } 2884 2885 /* 2886 * Check if the chip is in an error state (UE'd) before proceeding. 2887 */ 2888 if (sli_fw_error_status(&hw->sli) > 0) { 2889 ocs_log_crit(hw->os, "Chip is in an error state - reset needed\n"); 2890 return OCS_HW_RTN_ERROR; 2891 } 2892 2893 ocs_sm_post_event(&domain->sm, OCS_EVT_HW_DOMAIN_REQ_FREE, NULL); 2894 return rc; 2895 } 2896 2897 /** 2898 * @ingroup domain 2899 * @brief Free a fabric domain object. 2900 * 2901 * @par Description 2902 * Free the driver resources associated with the domain. The difference between 2903 * this call and ocs_hw_domain_free() is that this call assumes resources no longer 2904 * exist on the SLI port, due to a reset or after some error conditions. 2905 * 2906 * @param hw Hardware context. 2907 * @param domain Pointer to the domain object. 2908 * 2909 * @return Returns 0 on success, or a non-zero value on failure. 2910 */ 2911 ocs_hw_rtn_e 2912 ocs_hw_domain_force_free(ocs_hw_t *hw, ocs_domain_t *domain) 2913 { 2914 if (!hw || !domain) { 2915 ocs_log_err(NULL, "bad parameter(s) hw=%p domain=%p\n", hw, domain); 2916 return OCS_HW_RTN_ERROR; 2917 } 2918 2919 sli_resource_free(&hw->sli, SLI_RSRC_FCOE_VFI, domain->indicator); 2920 2921 return OCS_HW_RTN_SUCCESS; 2922 } 2923 2924 /** 2925 * @ingroup node 2926 * @brief Allocate a remote node object. 2927 * 2928 * @param hw Hardware context. 2929 * @param rnode Allocated remote node object to initialize. 2930 * @param fc_addr FC address of the remote node. 2931 * @param sport SLI port used to connect to remote node. 2932 * 2933 * @return Returns 0 on success, or a non-zero value on failure. 2934 */ 2935 ocs_hw_rtn_e 2936 ocs_hw_node_alloc(ocs_hw_t *hw, ocs_remote_node_t *rnode, uint32_t fc_addr, 2937 ocs_sli_port_t *sport) 2938 { 2939 /* Check for invalid indicator */ 2940 if (UINT32_MAX != rnode->indicator) { 2941 ocs_log_err(hw->os, "FCOE_RPI allocation failure addr=%#x rpi=%#x\n", 2942 fc_addr, rnode->indicator); 2943 return OCS_HW_RTN_ERROR; 2944 } 2945 2946 /* 2947 * Check if the chip is in an error state (UE'd) before proceeding. 2948 */ 2949 if (sli_fw_error_status(&hw->sli) > 0) { 2950 ocs_log_crit(hw->os, "Chip is in an error state - reset needed\n"); 2951 return OCS_HW_RTN_ERROR; 2952 } 2953 2954 /* NULL SLI port indicates an unallocated remote node */ 2955 rnode->sport = NULL; 2956 2957 if (sli_resource_alloc(&hw->sli, SLI_RSRC_FCOE_RPI, &rnode->indicator, &rnode->index)) { 2958 ocs_log_err(hw->os, "FCOE_RPI allocation failure addr=%#x\n", 2959 fc_addr); 2960 return OCS_HW_RTN_ERROR; 2961 } 2962 2963 rnode->fc_id = fc_addr; 2964 rnode->sport = sport; 2965 2966 return OCS_HW_RTN_SUCCESS; 2967 } 2968 2969 /** 2970 * @ingroup node 2971 * @brief Update a remote node object with the remote port's service parameters. 2972 * 2973 * @param hw Hardware context. 2974 * @param rnode Allocated remote node object to initialize. 2975 * @param sparms DMA buffer containing the remote port's service parameters. 2976 * 2977 * @return Returns 0 on success, or a non-zero value on failure. 2978 */ 2979 ocs_hw_rtn_e 2980 ocs_hw_node_attach(ocs_hw_t *hw, ocs_remote_node_t *rnode, ocs_dma_t *sparms) 2981 { 2982 ocs_hw_rtn_e rc = OCS_HW_RTN_ERROR; 2983 uint8_t *buf = NULL; 2984 uint32_t count = 0; 2985 2986 if (!hw || !rnode || !sparms) { 2987 ocs_log_err(NULL, "bad parameter(s) hw=%p rnode=%p sparms=%p\n", 2988 hw, rnode, sparms); 2989 return OCS_HW_RTN_ERROR; 2990 } 2991 2992 /* 2993 * Check if the chip is in an error state (UE'd) before proceeding. 2994 */ 2995 if (sli_fw_error_status(&hw->sli) > 0) { 2996 ocs_log_crit(hw->os, "Chip is in an error state - reset needed\n"); 2997 return OCS_HW_RTN_ERROR; 2998 } 2999 3000 buf = ocs_malloc(hw->os, SLI4_BMBX_SIZE, OCS_M_NOWAIT); 3001 if (!buf) { 3002 ocs_log_err(hw->os, "no buffer for command\n"); 3003 return OCS_HW_RTN_NO_MEMORY; 3004 } 3005 3006 /* 3007 * If the attach count is non-zero, this RPI has already been registered. 3008 * Otherwise, register the RPI 3009 */ 3010 if (rnode->index == UINT32_MAX) { 3011 ocs_log_err(NULL, "bad parameter rnode->index invalid\n"); 3012 ocs_free(hw->os, buf, SLI4_BMBX_SIZE); 3013 return OCS_HW_RTN_ERROR; 3014 } 3015 count = ocs_atomic_add_return(&hw->rpi_ref[rnode->index].rpi_count, 1); 3016 if (count) { 3017 /* 3018 * Can't attach multiple FC_ID's to a node unless High Login 3019 * Mode is enabled 3020 */ 3021 if (sli_get_hlm(&hw->sli) == FALSE) { 3022 ocs_log_test(hw->os, "attach to already attached node HLM=%d count=%d\n", 3023 sli_get_hlm(&hw->sli), count); 3024 rc = OCS_HW_RTN_SUCCESS; 3025 } else { 3026 rnode->node_group = TRUE; 3027 rnode->attached = ocs_atomic_read(&hw->rpi_ref[rnode->index].rpi_attached); 3028 rc = rnode->attached ? OCS_HW_RTN_SUCCESS_SYNC : OCS_HW_RTN_SUCCESS; 3029 } 3030 } else { 3031 rnode->node_group = FALSE; 3032 3033 ocs_display_sparams("", "reg rpi", 0, NULL, sparms->virt); 3034 if (sli_cmd_reg_rpi(&hw->sli, buf, SLI4_BMBX_SIZE, rnode->fc_id, 3035 rnode->indicator, rnode->sport->indicator, 3036 sparms, 0, (hw->auto_xfer_rdy_enabled && hw->config.auto_xfer_rdy_t10_enable))) { 3037 rc = ocs_hw_command(hw, buf, OCS_CMD_NOWAIT, 3038 ocs_hw_cb_node_attach, rnode); 3039 } 3040 } 3041 3042 if (count || rc) { 3043 if (rc < OCS_HW_RTN_SUCCESS) { 3044 ocs_atomic_sub_return(&hw->rpi_ref[rnode->index].rpi_count, 1); 3045 ocs_log_err(hw->os, "%s error\n", count ? "HLM" : "REG_RPI"); 3046 } 3047 ocs_free(hw->os, buf, SLI4_BMBX_SIZE); 3048 } 3049 3050 return rc; 3051 } 3052 3053 /** 3054 * @ingroup node 3055 * @brief Free a remote node resource. 3056 * 3057 * @param hw Hardware context. 3058 * @param rnode Remote node object to free. 3059 * 3060 * @return Returns 0 on success, or a non-zero value on failure. 3061 */ 3062 ocs_hw_rtn_e 3063 ocs_hw_node_free_resources(ocs_hw_t *hw, ocs_remote_node_t *rnode) 3064 { 3065 ocs_hw_rtn_e rc = OCS_HW_RTN_SUCCESS; 3066 3067 if (!hw || !rnode) { 3068 ocs_log_err(NULL, "bad parameter(s) hw=%p rnode=%p\n", 3069 hw, rnode); 3070 return OCS_HW_RTN_ERROR; 3071 } 3072 3073 if (rnode->sport) { 3074 if (!rnode->attached) { 3075 if (rnode->indicator != UINT32_MAX) { 3076 if (sli_resource_free(&hw->sli, SLI_RSRC_FCOE_RPI, rnode->indicator)) { 3077 ocs_log_err(hw->os, "FCOE_RPI free failure RPI %d addr=%#x\n", 3078 rnode->indicator, rnode->fc_id); 3079 rc = OCS_HW_RTN_ERROR; 3080 } else { 3081 rnode->node_group = FALSE; 3082 rnode->indicator = UINT32_MAX; 3083 rnode->index = UINT32_MAX; 3084 rnode->free_group = FALSE; 3085 } 3086 } 3087 } else { 3088 ocs_log_err(hw->os, "Error: rnode is still attached\n"); 3089 rc = OCS_HW_RTN_ERROR; 3090 } 3091 } 3092 3093 return rc; 3094 } 3095 3096 /** 3097 * @ingroup node 3098 * @brief Free a remote node object. 3099 * 3100 * @param hw Hardware context. 3101 * @param rnode Remote node object to free. 3102 * 3103 * @return Returns 0 on success, or a non-zero value on failure. 3104 */ 3105 ocs_hw_rtn_e 3106 ocs_hw_node_detach(ocs_hw_t *hw, ocs_remote_node_t *rnode) 3107 { 3108 uint8_t *buf = NULL; 3109 ocs_hw_rtn_e rc = OCS_HW_RTN_SUCCESS_SYNC; 3110 uint32_t index = UINT32_MAX; 3111 3112 if (!hw || !rnode) { 3113 ocs_log_err(NULL, "bad parameter(s) hw=%p rnode=%p\n", 3114 hw, rnode); 3115 return OCS_HW_RTN_ERROR; 3116 } 3117 3118 /* 3119 * Check if the chip is in an error state (UE'd) before proceeding. 3120 */ 3121 if (sli_fw_error_status(&hw->sli) > 0) { 3122 ocs_log_crit(hw->os, "Chip is in an error state - reset needed\n"); 3123 return OCS_HW_RTN_ERROR; 3124 } 3125 3126 index = rnode->index; 3127 3128 if (rnode->sport) { 3129 uint32_t count = 0; 3130 uint32_t fc_id; 3131 3132 if (!rnode->attached) { 3133 return OCS_HW_RTN_SUCCESS_SYNC; 3134 } 3135 3136 buf = ocs_malloc(hw->os, SLI4_BMBX_SIZE, OCS_M_NOWAIT); 3137 if (!buf) { 3138 ocs_log_err(hw->os, "no buffer for command\n"); 3139 return OCS_HW_RTN_NO_MEMORY; 3140 } 3141 3142 count = ocs_atomic_sub_return(&hw->rpi_ref[index].rpi_count, 1); 3143 3144 if (count <= 1) { 3145 /* There are no other references to this RPI 3146 * so unregister it and free the resource. */ 3147 fc_id = UINT32_MAX; 3148 rnode->node_group = FALSE; 3149 rnode->free_group = TRUE; 3150 } else { 3151 if (sli_get_hlm(&hw->sli) == FALSE) { 3152 ocs_log_test(hw->os, "Invalid count with HLM disabled, count=%d\n", 3153 count); 3154 } 3155 fc_id = rnode->fc_id & 0x00ffffff; 3156 } 3157 3158 rc = OCS_HW_RTN_ERROR; 3159 3160 if (sli_cmd_unreg_rpi(&hw->sli, buf, SLI4_BMBX_SIZE, rnode->indicator, 3161 SLI_RSRC_FCOE_RPI, fc_id)) { 3162 rc = ocs_hw_command(hw, buf, OCS_CMD_NOWAIT, ocs_hw_cb_node_free, rnode); 3163 } 3164 3165 if (rc != OCS_HW_RTN_SUCCESS) { 3166 ocs_log_err(hw->os, "UNREG_RPI failed\n"); 3167 ocs_free(hw->os, buf, SLI4_BMBX_SIZE); 3168 rc = OCS_HW_RTN_ERROR; 3169 } 3170 } 3171 3172 return rc; 3173 } 3174 3175 /** 3176 * @ingroup node 3177 * @brief Free all remote node objects. 3178 * 3179 * @param hw Hardware context. 3180 * 3181 * @return Returns 0 on success, or a non-zero value on failure. 3182 */ 3183 ocs_hw_rtn_e 3184 ocs_hw_node_free_all(ocs_hw_t *hw) 3185 { 3186 uint8_t *buf = NULL; 3187 ocs_hw_rtn_e rc = OCS_HW_RTN_ERROR; 3188 3189 if (!hw) { 3190 ocs_log_err(NULL, "bad parameter hw=%p\n", hw); 3191 return OCS_HW_RTN_ERROR; 3192 } 3193 3194 /* 3195 * Check if the chip is in an error state (UE'd) before proceeding. 3196 */ 3197 if (sli_fw_error_status(&hw->sli) > 0) { 3198 ocs_log_crit(hw->os, "Chip is in an error state - reset needed\n"); 3199 return OCS_HW_RTN_ERROR; 3200 } 3201 3202 buf = ocs_malloc(hw->os, SLI4_BMBX_SIZE, OCS_M_NOWAIT); 3203 if (!buf) { 3204 ocs_log_err(hw->os, "no buffer for command\n"); 3205 return OCS_HW_RTN_NO_MEMORY; 3206 } 3207 3208 if (sli_cmd_unreg_rpi(&hw->sli, buf, SLI4_BMBX_SIZE, 0xffff, 3209 SLI_RSRC_FCOE_FCFI, UINT32_MAX)) { 3210 rc = ocs_hw_command(hw, buf, OCS_CMD_NOWAIT, ocs_hw_cb_node_free_all, 3211 NULL); 3212 } 3213 3214 if (rc != OCS_HW_RTN_SUCCESS) { 3215 ocs_log_err(hw->os, "UNREG_RPI failed\n"); 3216 ocs_free(hw->os, buf, SLI4_BMBX_SIZE); 3217 rc = OCS_HW_RTN_ERROR; 3218 } 3219 3220 return rc; 3221 } 3222 3223 ocs_hw_rtn_e 3224 ocs_hw_node_group_alloc(ocs_hw_t *hw, ocs_remote_node_group_t *ngroup) 3225 { 3226 3227 if (!hw || !ngroup) { 3228 ocs_log_err(NULL, "bad parameter hw=%p ngroup=%p\n", 3229 hw, ngroup); 3230 return OCS_HW_RTN_ERROR; 3231 } 3232 3233 if (sli_resource_alloc(&hw->sli, SLI_RSRC_FCOE_RPI, &ngroup->indicator, 3234 &ngroup->index)) { 3235 ocs_log_err(hw->os, "FCOE_RPI allocation failure addr=%#x\n", 3236 ngroup->indicator); 3237 return OCS_HW_RTN_ERROR; 3238 } 3239 3240 return OCS_HW_RTN_SUCCESS; 3241 } 3242 3243 ocs_hw_rtn_e 3244 ocs_hw_node_group_attach(ocs_hw_t *hw, ocs_remote_node_group_t *ngroup, ocs_remote_node_t *rnode) 3245 { 3246 3247 if (!hw || !ngroup || !rnode) { 3248 ocs_log_err(NULL, "bad parameter hw=%p ngroup=%p rnode=%p\n", 3249 hw, ngroup, rnode); 3250 return OCS_HW_RTN_ERROR; 3251 } 3252 3253 if (rnode->attached) { 3254 ocs_log_err(hw->os, "node already attached RPI=%#x addr=%#x\n", 3255 rnode->indicator, rnode->fc_id); 3256 return OCS_HW_RTN_ERROR; 3257 } 3258 3259 if (sli_resource_free(&hw->sli, SLI_RSRC_FCOE_RPI, rnode->indicator)) { 3260 ocs_log_err(hw->os, "FCOE_RPI free failure RPI=%#x\n", 3261 rnode->indicator); 3262 return OCS_HW_RTN_ERROR; 3263 } 3264 3265 rnode->indicator = ngroup->indicator; 3266 rnode->index = ngroup->index; 3267 3268 return OCS_HW_RTN_SUCCESS; 3269 } 3270 3271 ocs_hw_rtn_e 3272 ocs_hw_node_group_free(ocs_hw_t *hw, ocs_remote_node_group_t *ngroup) 3273 { 3274 int ref; 3275 3276 if (!hw || !ngroup) { 3277 ocs_log_err(NULL, "bad parameter hw=%p ngroup=%p\n", 3278 hw, ngroup); 3279 return OCS_HW_RTN_ERROR; 3280 } 3281 3282 ref = ocs_atomic_read(&hw->rpi_ref[ngroup->index].rpi_count); 3283 if (ref) { 3284 /* Hmmm, the reference count is non-zero */ 3285 ocs_log_debug(hw->os, "node group reference=%d (RPI=%#x)\n", 3286 ref, ngroup->indicator); 3287 3288 if (sli_resource_free(&hw->sli, SLI_RSRC_FCOE_RPI, ngroup->indicator)) { 3289 ocs_log_err(hw->os, "FCOE_RPI free failure RPI=%#x\n", 3290 ngroup->indicator); 3291 return OCS_HW_RTN_ERROR; 3292 } 3293 3294 ocs_atomic_set(&hw->rpi_ref[ngroup->index].rpi_count, 0); 3295 } 3296 3297 ngroup->indicator = UINT32_MAX; 3298 ngroup->index = UINT32_MAX; 3299 3300 return OCS_HW_RTN_SUCCESS; 3301 } 3302 3303 /** 3304 * @brief Initialize IO fields on each free call. 3305 * 3306 * @n @b Note: This is done on each free call (as opposed to each 3307 * alloc call) because port-owned XRIs are not 3308 * allocated with ocs_hw_io_alloc() but are freed with this 3309 * function. 3310 * 3311 * @param io Pointer to HW IO. 3312 */ 3313 static inline void 3314 ocs_hw_init_free_io(ocs_hw_io_t *io) 3315 { 3316 /* 3317 * Set io->done to NULL, to avoid any callbacks, should 3318 * a completion be received for one of these IOs 3319 */ 3320 io->done = NULL; 3321 io->abort_done = NULL; 3322 io->status_saved = 0; 3323 io->abort_in_progress = FALSE; 3324 io->port_owned_abort_count = 0; 3325 io->rnode = NULL; 3326 io->type = 0xFFFF; 3327 io->wq = NULL; 3328 io->ul_io = NULL; 3329 io->tgt_wqe_timeout = 0; 3330 } 3331 3332 /** 3333 * @ingroup io 3334 * @brief Lockless allocate a HW IO object. 3335 * 3336 * @par Description 3337 * Assume that hw->ocs_lock is held. This function is only used if 3338 * use_dif_sec_xri workaround is being used. 3339 * 3340 * @param hw Hardware context. 3341 * 3342 * @return Returns a pointer to an object on success, or NULL on failure. 3343 */ 3344 static inline ocs_hw_io_t * 3345 _ocs_hw_io_alloc(ocs_hw_t *hw) 3346 { 3347 ocs_hw_io_t *io = NULL; 3348 3349 if (NULL != (io = ocs_list_remove_head(&hw->io_free))) { 3350 ocs_list_add_tail(&hw->io_inuse, io); 3351 io->state = OCS_HW_IO_STATE_INUSE; 3352 io->quarantine = FALSE; 3353 io->quarantine_first_phase = TRUE; 3354 io->abort_reqtag = UINT32_MAX; 3355 ocs_ref_init(&io->ref, ocs_hw_io_free_internal, io); 3356 } else { 3357 ocs_atomic_add_return(&hw->io_alloc_failed_count, 1); 3358 } 3359 3360 return io; 3361 } 3362 /** 3363 * @ingroup io 3364 * @brief Allocate a HW IO object. 3365 * 3366 * @par Description 3367 * @n @b Note: This function applies to non-port owned XRIs 3368 * only. 3369 * 3370 * @param hw Hardware context. 3371 * 3372 * @return Returns a pointer to an object on success, or NULL on failure. 3373 */ 3374 ocs_hw_io_t * 3375 ocs_hw_io_alloc(ocs_hw_t *hw) 3376 { 3377 ocs_hw_io_t *io = NULL; 3378 3379 ocs_lock(&hw->io_lock); 3380 io = _ocs_hw_io_alloc(hw); 3381 ocs_unlock(&hw->io_lock); 3382 3383 return io; 3384 } 3385 3386 /** 3387 * @ingroup io 3388 * @brief Allocate/Activate a port owned HW IO object. 3389 * 3390 * @par Description 3391 * This function is called by the transport layer when an XRI is 3392 * allocated by the SLI-Port. This will "activate" the HW IO 3393 * associated with the XRI received from the SLI-Port to mirror 3394 * the state of the XRI. 3395 * @n @n @b Note: This function applies to port owned XRIs only. 3396 * 3397 * @param hw Hardware context. 3398 * @param io Pointer HW IO to activate/allocate. 3399 * 3400 * @return Returns a pointer to an object on success, or NULL on failure. 3401 */ 3402 ocs_hw_io_t * 3403 ocs_hw_io_activate_port_owned(ocs_hw_t *hw, ocs_hw_io_t *io) 3404 { 3405 if (ocs_ref_read_count(&io->ref) > 0) { 3406 ocs_log_err(hw->os, "Bad parameter: refcount > 0\n"); 3407 return NULL; 3408 } 3409 3410 if (io->wq != NULL) { 3411 ocs_log_err(hw->os, "XRI %x already in use\n", io->indicator); 3412 return NULL; 3413 } 3414 3415 ocs_ref_init(&io->ref, ocs_hw_io_free_port_owned, io); 3416 io->xbusy = TRUE; 3417 3418 return io; 3419 } 3420 3421 /** 3422 * @ingroup io 3423 * @brief When an IO is freed, depending on the exchange busy flag, and other 3424 * workarounds, move it to the correct list. 3425 * 3426 * @par Description 3427 * @n @b Note: Assumes that the hw->io_lock is held and the item has been removed 3428 * from the busy or wait_free list. 3429 * 3430 * @param hw Hardware context. 3431 * @param io Pointer to the IO object to move. 3432 */ 3433 static void 3434 ocs_hw_io_free_move_correct_list(ocs_hw_t *hw, ocs_hw_io_t *io) 3435 { 3436 if (io->xbusy) { 3437 /* add to wait_free list and wait for XRI_ABORTED CQEs to clean up */ 3438 ocs_list_add_tail(&hw->io_wait_free, io); 3439 io->state = OCS_HW_IO_STATE_WAIT_FREE; 3440 } else { 3441 /* IO not busy, add to free list */ 3442 ocs_list_add_tail(&hw->io_free, io); 3443 io->state = OCS_HW_IO_STATE_FREE; 3444 } 3445 3446 /* BZ 161832 workaround */ 3447 if (hw->workaround.use_dif_sec_xri) { 3448 ocs_hw_check_sec_hio_list(hw); 3449 } 3450 } 3451 3452 /** 3453 * @ingroup io 3454 * @brief Free a HW IO object. Perform cleanup common to 3455 * port and host-owned IOs. 3456 * 3457 * @param hw Hardware context. 3458 * @param io Pointer to the HW IO object. 3459 */ 3460 static inline void 3461 ocs_hw_io_free_common(ocs_hw_t *hw, ocs_hw_io_t *io) 3462 { 3463 /* initialize IO fields */ 3464 ocs_hw_init_free_io(io); 3465 3466 /* Restore default SGL */ 3467 ocs_hw_io_restore_sgl(hw, io); 3468 } 3469 3470 /** 3471 * @ingroup io 3472 * @brief Free a HW IO object associated with a port-owned XRI. 3473 * 3474 * @param arg Pointer to the HW IO object. 3475 */ 3476 static void 3477 ocs_hw_io_free_port_owned(void *arg) 3478 { 3479 ocs_hw_io_t *io = (ocs_hw_io_t *)arg; 3480 ocs_hw_t *hw = io->hw; 3481 3482 /* 3483 * For auto xfer rdy, if the dnrx bit is set, then add it to the list of XRIs 3484 * waiting for buffers. 3485 */ 3486 if (io->auto_xfer_rdy_dnrx) { 3487 ocs_lock(&hw->io_lock); 3488 /* take a reference count because we still own the IO until the buffer is posted */ 3489 ocs_ref_init(&io->ref, ocs_hw_io_free_port_owned, io); 3490 ocs_list_add_tail(&hw->io_port_dnrx, io); 3491 ocs_unlock(&hw->io_lock); 3492 } 3493 3494 /* perform common cleanup */ 3495 ocs_hw_io_free_common(hw, io); 3496 } 3497 3498 /** 3499 * @ingroup io 3500 * @brief Free a previously-allocated HW IO object. Called when 3501 * IO refcount goes to zero (host-owned IOs only). 3502 * 3503 * @param arg Pointer to the HW IO object. 3504 */ 3505 static void 3506 ocs_hw_io_free_internal(void *arg) 3507 { 3508 ocs_hw_io_t *io = (ocs_hw_io_t *)arg; 3509 ocs_hw_t *hw = io->hw; 3510 3511 /* perform common cleanup */ 3512 ocs_hw_io_free_common(hw, io); 3513 3514 ocs_lock(&hw->io_lock); 3515 /* remove from in-use list */ 3516 ocs_list_remove(&hw->io_inuse, io); 3517 ocs_hw_io_free_move_correct_list(hw, io); 3518 ocs_unlock(&hw->io_lock); 3519 } 3520 3521 /** 3522 * @ingroup io 3523 * @brief Free a previously-allocated HW IO object. 3524 * 3525 * @par Description 3526 * @n @b Note: This function applies to port and host owned XRIs. 3527 * 3528 * @param hw Hardware context. 3529 * @param io Pointer to the HW IO object. 3530 * 3531 * @return Returns a non-zero value if HW IO was freed, 0 if references 3532 * on the IO still exist, or a negative value if an error occurred. 3533 */ 3534 int32_t 3535 ocs_hw_io_free(ocs_hw_t *hw, ocs_hw_io_t *io) 3536 { 3537 /* just put refcount */ 3538 if (ocs_ref_read_count(&io->ref) <= 0) { 3539 ocs_log_err(hw->os, "Bad parameter: refcount <= 0 xri=%x tag=%x\n", 3540 io->indicator, io->reqtag); 3541 return -1; 3542 } 3543 3544 return ocs_ref_put(&io->ref); /* ocs_ref_get(): ocs_hw_io_alloc() */ 3545 } 3546 3547 /** 3548 * @ingroup io 3549 * @brief Check if given HW IO is in-use 3550 * 3551 * @par Description 3552 * This function returns TRUE if the given HW IO has been 3553 * allocated and is in-use, and FALSE otherwise. It applies to 3554 * port and host owned XRIs. 3555 * 3556 * @param hw Hardware context. 3557 * @param io Pointer to the HW IO object. 3558 * 3559 * @return TRUE if an IO is in use, or FALSE otherwise. 3560 */ 3561 uint8_t 3562 ocs_hw_io_inuse(ocs_hw_t *hw, ocs_hw_io_t *io) 3563 { 3564 return (ocs_ref_read_count(&io->ref) > 0); 3565 } 3566 3567 /** 3568 * @brief Write a HW IO to a work queue. 3569 * 3570 * @par Description 3571 * A HW IO is written to a work queue. 3572 * 3573 * @param wq Pointer to work queue. 3574 * @param wqe Pointer to WQ entry. 3575 * 3576 * @n @b Note: Assumes the SLI-4 queue lock is held. 3577 * 3578 * @return Returns 0 on success, or a negative error code value on failure. 3579 */ 3580 static int32_t 3581 _hw_wq_write(hw_wq_t *wq, ocs_hw_wqe_t *wqe) 3582 { 3583 int32_t rc; 3584 int32_t queue_rc; 3585 3586 /* Every so often, set the wqec bit to generate comsummed completions */ 3587 if (wq->wqec_count) { 3588 wq->wqec_count--; 3589 } 3590 if (wq->wqec_count == 0) { 3591 sli4_generic_wqe_t *genwqe = (void*)wqe->wqebuf; 3592 genwqe->wqec = 1; 3593 wq->wqec_count = wq->wqec_set_count; 3594 } 3595 3596 /* Decrement WQ free count */ 3597 wq->free_count--; 3598 3599 queue_rc = _sli_queue_write(&wq->hw->sli, wq->queue, wqe->wqebuf); 3600 3601 if (queue_rc < 0) { 3602 rc = -1; 3603 } else { 3604 rc = 0; 3605 ocs_queue_history_wq(&wq->hw->q_hist, (void *) wqe->wqebuf, wq->queue->id, queue_rc); 3606 } 3607 3608 return rc; 3609 } 3610 3611 /** 3612 * @brief Write a HW IO to a work queue. 3613 * 3614 * @par Description 3615 * A HW IO is written to a work queue. 3616 * 3617 * @param wq Pointer to work queue. 3618 * @param wqe Pointer to WQE entry. 3619 * 3620 * @n @b Note: Takes the SLI-4 queue lock. 3621 * 3622 * @return Returns 0 on success, or a negative error code value on failure. 3623 */ 3624 int32_t 3625 hw_wq_write(hw_wq_t *wq, ocs_hw_wqe_t *wqe) 3626 { 3627 int32_t rc = 0; 3628 3629 sli_queue_lock(wq->queue); 3630 if ( ! ocs_list_empty(&wq->pending_list)) { 3631 ocs_list_add_tail(&wq->pending_list, wqe); 3632 OCS_STAT(wq->wq_pending_count++;) 3633 while ((wq->free_count > 0) && ((wqe = ocs_list_remove_head(&wq->pending_list)) != NULL)) { 3634 rc = _hw_wq_write(wq, wqe); 3635 if (rc < 0) { 3636 break; 3637 } 3638 if (wqe->abort_wqe_submit_needed) { 3639 wqe->abort_wqe_submit_needed = 0; 3640 sli_abort_wqe(&wq->hw->sli, wqe->wqebuf, wq->hw->sli.config.wqe_size, SLI_ABORT_XRI, 3641 wqe->send_abts, wqe->id, 0, wqe->abort_reqtag, SLI4_CQ_DEFAULT ); 3642 ocs_list_add_tail(&wq->pending_list, wqe); 3643 OCS_STAT(wq->wq_pending_count++;) 3644 } 3645 } 3646 } else { 3647 if (wq->free_count > 0) { 3648 rc = _hw_wq_write(wq, wqe); 3649 } else { 3650 ocs_list_add_tail(&wq->pending_list, wqe); 3651 OCS_STAT(wq->wq_pending_count++;) 3652 } 3653 } 3654 3655 sli_queue_unlock(wq->queue); 3656 3657 return rc; 3658 3659 } 3660 3661 /** 3662 * @brief Update free count and submit any pending HW IOs 3663 * 3664 * @par Description 3665 * The WQ free count is updated, and any pending HW IOs are submitted that 3666 * will fit in the queue. 3667 * 3668 * @param wq Pointer to work queue. 3669 * @param update_free_count Value added to WQs free count. 3670 * 3671 * @return None. 3672 */ 3673 static void 3674 hw_wq_submit_pending(hw_wq_t *wq, uint32_t update_free_count) 3675 { 3676 ocs_hw_wqe_t *wqe; 3677 3678 sli_queue_lock(wq->queue); 3679 3680 /* Update free count with value passed in */ 3681 wq->free_count += update_free_count; 3682 3683 while ((wq->free_count > 0) && ((wqe = ocs_list_remove_head(&wq->pending_list)) != NULL)) { 3684 _hw_wq_write(wq, wqe); 3685 3686 if (wqe->abort_wqe_submit_needed) { 3687 wqe->abort_wqe_submit_needed = 0; 3688 sli_abort_wqe(&wq->hw->sli, wqe->wqebuf, wq->hw->sli.config.wqe_size, SLI_ABORT_XRI, 3689 wqe->send_abts, wqe->id, 0, wqe->abort_reqtag, SLI4_CQ_DEFAULT); 3690 ocs_list_add_tail(&wq->pending_list, wqe); 3691 OCS_STAT(wq->wq_pending_count++;) 3692 } 3693 } 3694 3695 sli_queue_unlock(wq->queue); 3696 } 3697 3698 /** 3699 * @brief Check to see if there are any BZ 161832 workaround waiting IOs 3700 * 3701 * @par Description 3702 * Checks hw->sec_hio_wait_list, if an IO is waiting for a HW IO, then try 3703 * to allocate a secondary HW io, and dispatch it. 3704 * 3705 * @n @b Note: hw->io_lock MUST be taken when called. 3706 * 3707 * @param hw pointer to HW object 3708 * 3709 * @return none 3710 */ 3711 static void 3712 ocs_hw_check_sec_hio_list(ocs_hw_t *hw) 3713 { 3714 ocs_hw_io_t *io; 3715 ocs_hw_io_t *sec_io; 3716 int rc = 0; 3717 3718 while (!ocs_list_empty(&hw->sec_hio_wait_list)) { 3719 uint16_t flags; 3720 3721 sec_io = _ocs_hw_io_alloc(hw); 3722 if (sec_io == NULL) { 3723 break; 3724 } 3725 3726 io = ocs_list_remove_head(&hw->sec_hio_wait_list); 3727 ocs_list_add_tail(&hw->io_inuse, io); 3728 io->state = OCS_HW_IO_STATE_INUSE; 3729 io->sec_hio = sec_io; 3730 3731 /* mark secondary XRI for second and subsequent data phase as quarantine */ 3732 if (io->xbusy) { 3733 sec_io->quarantine = TRUE; 3734 } 3735 3736 flags = io->sec_iparam.fcp_tgt.flags; 3737 if (io->xbusy) { 3738 flags |= SLI4_IO_CONTINUATION; 3739 } else { 3740 flags &= ~SLI4_IO_CONTINUATION; 3741 } 3742 3743 io->tgt_wqe_timeout = io->sec_iparam.fcp_tgt.timeout; 3744 3745 /* Complete (continue) TRECV IO */ 3746 if (io->xbusy) { 3747 if (sli_fcp_cont_treceive64_wqe(&hw->sli, io->wqe.wqebuf, hw->sli.config.wqe_size, &io->def_sgl, 3748 io->first_data_sge, 3749 io->sec_iparam.fcp_tgt.offset, io->sec_len, io->indicator, io->sec_hio->indicator, 3750 io->reqtag, SLI4_CQ_DEFAULT, 3751 io->sec_iparam.fcp_tgt.ox_id, io->rnode->indicator, io->rnode, 3752 flags, 3753 io->sec_iparam.fcp_tgt.dif_oper, io->sec_iparam.fcp_tgt.blk_size, io->sec_iparam.fcp_tgt.cs_ctl, io->sec_iparam.fcp_tgt.app_id)) { 3754 ocs_log_test(hw->os, "TRECEIVE WQE error\n"); 3755 break; 3756 } 3757 } else { 3758 if (sli_fcp_treceive64_wqe(&hw->sli, io->wqe.wqebuf, hw->sli.config.wqe_size, &io->def_sgl, 3759 io->first_data_sge, 3760 io->sec_iparam.fcp_tgt.offset, io->sec_len, io->indicator, 3761 io->reqtag, SLI4_CQ_DEFAULT, 3762 io->sec_iparam.fcp_tgt.ox_id, io->rnode->indicator, io->rnode, 3763 flags, 3764 io->sec_iparam.fcp_tgt.dif_oper, io->sec_iparam.fcp_tgt.blk_size, 3765 io->sec_iparam.fcp_tgt.cs_ctl, io->sec_iparam.fcp_tgt.app_id)) { 3766 ocs_log_test(hw->os, "TRECEIVE WQE error\n"); 3767 break; 3768 } 3769 } 3770 3771 if (io->wq == NULL) { 3772 io->wq = ocs_hw_queue_next_wq(hw, io); 3773 ocs_hw_assert(io->wq != NULL); 3774 } 3775 io->xbusy = TRUE; 3776 3777 /* 3778 * Add IO to active io wqe list before submitting, in case the 3779 * wcqe processing preempts this thread. 3780 */ 3781 ocs_hw_add_io_timed_wqe(hw, io); 3782 rc = hw_wq_write(io->wq, &io->wqe); 3783 if (rc >= 0) { 3784 /* non-negative return is success */ 3785 rc = 0; 3786 } else { 3787 /* failed to write wqe, remove from active wqe list */ 3788 ocs_log_err(hw->os, "sli_queue_write failed: %d\n", rc); 3789 io->xbusy = FALSE; 3790 ocs_hw_remove_io_timed_wqe(hw, io); 3791 } 3792 } 3793 } 3794 3795 /** 3796 * @ingroup io 3797 * @brief Send a Single Request/Response Sequence (SRRS). 3798 * 3799 * @par Description 3800 * This routine supports communication sequences consisting of a single 3801 * request and single response between two endpoints. Examples include: 3802 * - Sending an ELS request. 3803 * - Sending an ELS response - To send an ELS reponse, the caller must provide 3804 * the OX_ID from the received request. 3805 * - Sending a FC Common Transport (FC-CT) request - To send a FC-CT request, 3806 * the caller must provide the R_CTL, TYPE, and DF_CTL 3807 * values to place in the FC frame header. 3808 * . 3809 * @n @b Note: The caller is expected to provide both send and receive 3810 * buffers for requests. In the case of sending a response, no receive buffer 3811 * is necessary and the caller may pass in a NULL pointer. 3812 * 3813 * @param hw Hardware context. 3814 * @param type Type of sequence (ELS request/response, FC-CT). 3815 * @param io Previously-allocated HW IO object. 3816 * @param send DMA memory holding data to send (for example, ELS request, BLS response). 3817 * @param len Length, in bytes, of data to send. 3818 * @param receive Optional DMA memory to hold a response. 3819 * @param rnode Destination of data (that is, a remote node). 3820 * @param iparam IO parameters (ELS response and FC-CT). 3821 * @param cb Function call upon completion of sending the data (may be NULL). 3822 * @param arg Argument to pass to IO completion function. 3823 * 3824 * @return Returns 0 on success, or a non-zero on failure. 3825 */ 3826 ocs_hw_rtn_e 3827 ocs_hw_srrs_send(ocs_hw_t *hw, ocs_hw_io_type_e type, ocs_hw_io_t *io, 3828 ocs_dma_t *send, uint32_t len, ocs_dma_t *receive, 3829 ocs_remote_node_t *rnode, ocs_hw_io_param_t *iparam, 3830 ocs_hw_srrs_cb_t cb, void *arg) 3831 { 3832 sli4_sge_t *sge = NULL; 3833 ocs_hw_rtn_e rc = OCS_HW_RTN_SUCCESS; 3834 uint16_t local_flags = 0; 3835 3836 if (!hw || !io || !rnode || !iparam) { 3837 ocs_log_err(NULL, "bad parm hw=%p io=%p send=%p receive=%p rnode=%p iparam=%p\n", 3838 hw, io, send, receive, rnode, iparam); 3839 return OCS_HW_RTN_ERROR; 3840 } 3841 3842 if (hw->state != OCS_HW_STATE_ACTIVE) { 3843 ocs_log_test(hw->os, "cannot send SRRS, HW state=%d\n", hw->state); 3844 return OCS_HW_RTN_ERROR; 3845 } 3846 3847 if (ocs_hw_is_xri_port_owned(hw, io->indicator)) { 3848 /* We must set the XC bit for port owned XRIs */ 3849 local_flags |= SLI4_IO_CONTINUATION; 3850 } 3851 io->rnode = rnode; 3852 io->type = type; 3853 io->done = cb; 3854 io->arg = arg; 3855 3856 sge = io->sgl->virt; 3857 3858 /* clear both SGE */ 3859 ocs_memset(io->sgl->virt, 0, 2 * sizeof(sli4_sge_t)); 3860 3861 if (send) { 3862 sge[0].buffer_address_high = ocs_addr32_hi(send->phys); 3863 sge[0].buffer_address_low = ocs_addr32_lo(send->phys); 3864 sge[0].sge_type = SLI4_SGE_TYPE_DATA; 3865 sge[0].buffer_length = len; 3866 } 3867 3868 if ((OCS_HW_ELS_REQ == type) || (OCS_HW_FC_CT == type)) { 3869 sge[1].buffer_address_high = ocs_addr32_hi(receive->phys); 3870 sge[1].buffer_address_low = ocs_addr32_lo(receive->phys); 3871 sge[1].sge_type = SLI4_SGE_TYPE_DATA; 3872 sge[1].buffer_length = receive->size; 3873 sge[1].last = TRUE; 3874 } else { 3875 sge[0].last = TRUE; 3876 } 3877 3878 switch (type) { 3879 case OCS_HW_ELS_REQ: 3880 if ( (!send) || sli_els_request64_wqe(&hw->sli, io->wqe.wqebuf, hw->sli.config.wqe_size, io->sgl, 3881 *((uint8_t *)(send->virt)), /* req_type */ 3882 len, receive->size, 3883 iparam->els.timeout, io->indicator, io->reqtag, SLI4_CQ_DEFAULT, rnode)) { 3884 ocs_log_err(hw->os, "REQ WQE error\n"); 3885 rc = OCS_HW_RTN_ERROR; 3886 } 3887 break; 3888 case OCS_HW_ELS_RSP: 3889 if ( (!send) || sli_xmit_els_rsp64_wqe(&hw->sli, io->wqe.wqebuf, hw->sli.config.wqe_size, send, len, 3890 io->indicator, io->reqtag, SLI4_CQ_DEFAULT, 3891 iparam->els.ox_id, 3892 rnode, local_flags, UINT32_MAX)) { 3893 ocs_log_err(hw->os, "RSP WQE error\n"); 3894 rc = OCS_HW_RTN_ERROR; 3895 } 3896 break; 3897 case OCS_HW_ELS_RSP_SID: 3898 if ( (!send) || sli_xmit_els_rsp64_wqe(&hw->sli, io->wqe.wqebuf, hw->sli.config.wqe_size, send, len, 3899 io->indicator, io->reqtag, SLI4_CQ_DEFAULT, 3900 iparam->els_sid.ox_id, 3901 rnode, local_flags, iparam->els_sid.s_id)) { 3902 ocs_log_err(hw->os, "RSP (SID) WQE error\n"); 3903 rc = OCS_HW_RTN_ERROR; 3904 } 3905 break; 3906 case OCS_HW_FC_CT: 3907 if ( (!send) || sli_gen_request64_wqe(&hw->sli, io->wqe.wqebuf, hw->sli.config.wqe_size, io->sgl, len, 3908 receive->size, iparam->fc_ct.timeout, io->indicator, 3909 io->reqtag, SLI4_CQ_DEFAULT, rnode, iparam->fc_ct.r_ctl, 3910 iparam->fc_ct.type, iparam->fc_ct.df_ctl)) { 3911 ocs_log_err(hw->os, "GEN WQE error\n"); 3912 rc = OCS_HW_RTN_ERROR; 3913 } 3914 break; 3915 case OCS_HW_FC_CT_RSP: 3916 if ( (!send) || sli_xmit_sequence64_wqe(&hw->sli, io->wqe.wqebuf, hw->sli.config.wqe_size, io->sgl, len, 3917 iparam->fc_ct_rsp.timeout, iparam->fc_ct_rsp.ox_id, io->indicator, 3918 io->reqtag, rnode, iparam->fc_ct_rsp.r_ctl, 3919 iparam->fc_ct_rsp.type, iparam->fc_ct_rsp.df_ctl)) { 3920 ocs_log_err(hw->os, "XMIT SEQ WQE error\n"); 3921 rc = OCS_HW_RTN_ERROR; 3922 } 3923 break; 3924 case OCS_HW_BLS_ACC: 3925 case OCS_HW_BLS_RJT: 3926 { 3927 sli_bls_payload_t bls; 3928 3929 if (OCS_HW_BLS_ACC == type) { 3930 bls.type = SLI_BLS_ACC; 3931 ocs_memcpy(&bls.u.acc, iparam->bls.payload, sizeof(bls.u.acc)); 3932 } else { 3933 bls.type = SLI_BLS_RJT; 3934 ocs_memcpy(&bls.u.rjt, iparam->bls.payload, sizeof(bls.u.rjt)); 3935 } 3936 3937 bls.ox_id = iparam->bls.ox_id; 3938 bls.rx_id = iparam->bls.rx_id; 3939 3940 if (sli_xmit_bls_rsp64_wqe(&hw->sli, io->wqe.wqebuf, hw->sli.config.wqe_size, &bls, 3941 io->indicator, io->reqtag, 3942 SLI4_CQ_DEFAULT, 3943 rnode, UINT32_MAX)) { 3944 ocs_log_err(hw->os, "XMIT_BLS_RSP64 WQE error\n"); 3945 rc = OCS_HW_RTN_ERROR; 3946 } 3947 break; 3948 } 3949 case OCS_HW_BLS_ACC_SID: 3950 { 3951 sli_bls_payload_t bls; 3952 3953 bls.type = SLI_BLS_ACC; 3954 ocs_memcpy(&bls.u.acc, iparam->bls_sid.payload, sizeof(bls.u.acc)); 3955 3956 bls.ox_id = iparam->bls_sid.ox_id; 3957 bls.rx_id = iparam->bls_sid.rx_id; 3958 3959 if (sli_xmit_bls_rsp64_wqe(&hw->sli, io->wqe.wqebuf, hw->sli.config.wqe_size, &bls, 3960 io->indicator, io->reqtag, 3961 SLI4_CQ_DEFAULT, 3962 rnode, iparam->bls_sid.s_id)) { 3963 ocs_log_err(hw->os, "XMIT_BLS_RSP64 WQE SID error\n"); 3964 rc = OCS_HW_RTN_ERROR; 3965 } 3966 break; 3967 } 3968 case OCS_HW_BCAST: 3969 if ( (!send) || sli_xmit_bcast64_wqe(&hw->sli, io->wqe.wqebuf, hw->sli.config.wqe_size, send, len, 3970 iparam->bcast.timeout, io->indicator, io->reqtag, 3971 SLI4_CQ_DEFAULT, rnode, 3972 iparam->bcast.r_ctl, iparam->bcast.type, iparam->bcast.df_ctl)) { 3973 ocs_log_err(hw->os, "XMIT_BCAST64 WQE error\n"); 3974 rc = OCS_HW_RTN_ERROR; 3975 } 3976 break; 3977 default: 3978 ocs_log_err(hw->os, "bad SRRS type %#x\n", type); 3979 rc = OCS_HW_RTN_ERROR; 3980 } 3981 3982 if (OCS_HW_RTN_SUCCESS == rc) { 3983 if (io->wq == NULL) { 3984 io->wq = ocs_hw_queue_next_wq(hw, io); 3985 ocs_hw_assert(io->wq != NULL); 3986 } 3987 io->xbusy = TRUE; 3988 3989 /* 3990 * Add IO to active io wqe list before submitting, in case the 3991 * wcqe processing preempts this thread. 3992 */ 3993 OCS_STAT(io->wq->use_count++); 3994 ocs_hw_add_io_timed_wqe(hw, io); 3995 rc = hw_wq_write(io->wq, &io->wqe); 3996 if (rc >= 0) { 3997 /* non-negative return is success */ 3998 rc = 0; 3999 } else { 4000 /* failed to write wqe, remove from active wqe list */ 4001 ocs_log_err(hw->os, "sli_queue_write failed: %d\n", rc); 4002 io->xbusy = FALSE; 4003 ocs_hw_remove_io_timed_wqe(hw, io); 4004 } 4005 } 4006 4007 return rc; 4008 } 4009 4010 /** 4011 * @ingroup io 4012 * @brief Send a read, write, or response IO. 4013 * 4014 * @par Description 4015 * This routine supports sending a higher-level IO (for example, FCP) between two endpoints 4016 * as a target or initiator. Examples include: 4017 * - Sending read data and good response (target). 4018 * - Sending a response (target with no data or after receiving write data). 4019 * . 4020 * This routine assumes all IOs use the SGL associated with the HW IO. Prior to 4021 * calling this routine, the data should be loaded using ocs_hw_io_add_sge(). 4022 * 4023 * @param hw Hardware context. 4024 * @param type Type of IO (target read, target response, and so on). 4025 * @param io Previously-allocated HW IO object. 4026 * @param len Length, in bytes, of data to send. 4027 * @param iparam IO parameters. 4028 * @param rnode Destination of data (that is, a remote node). 4029 * @param cb Function call upon completion of sending data (may be NULL). 4030 * @param arg Argument to pass to IO completion function. 4031 * 4032 * @return Returns 0 on success, or a non-zero value on failure. 4033 * 4034 * @todo 4035 * - Support specifiying relative offset. 4036 * - Use a WQ other than 0. 4037 */ 4038 ocs_hw_rtn_e 4039 ocs_hw_io_send(ocs_hw_t *hw, ocs_hw_io_type_e type, ocs_hw_io_t *io, 4040 uint32_t len, ocs_hw_io_param_t *iparam, ocs_remote_node_t *rnode, 4041 void *cb, void *arg) 4042 { 4043 ocs_hw_rtn_e rc = OCS_HW_RTN_SUCCESS; 4044 uint32_t rpi; 4045 uint8_t send_wqe = TRUE; 4046 4047 CPUTRACE(""); 4048 4049 if (!hw || !io || !rnode || !iparam) { 4050 ocs_log_err(NULL, "bad parm hw=%p io=%p iparam=%p rnode=%p\n", 4051 hw, io, iparam, rnode); 4052 return OCS_HW_RTN_ERROR; 4053 } 4054 4055 if (hw->state != OCS_HW_STATE_ACTIVE) { 4056 ocs_log_err(hw->os, "cannot send IO, HW state=%d\n", hw->state); 4057 return OCS_HW_RTN_ERROR; 4058 } 4059 4060 rpi = rnode->indicator; 4061 4062 if (hw->workaround.use_unregistered_rpi && (rpi == UINT32_MAX)) { 4063 rpi = hw->workaround.unregistered_rid; 4064 ocs_log_test(hw->os, "using unregistered RPI: %d\n", rpi); 4065 } 4066 4067 /* 4068 * Save state needed during later stages 4069 */ 4070 io->rnode = rnode; 4071 io->type = type; 4072 io->done = cb; 4073 io->arg = arg; 4074 4075 /* 4076 * Format the work queue entry used to send the IO 4077 */ 4078 switch (type) { 4079 case OCS_HW_IO_INITIATOR_READ: 4080 /* 4081 * If use_dif_quarantine workaround is in effect, and dif_separates then mark the 4082 * initiator read IO for quarantine 4083 */ 4084 if (hw->workaround.use_dif_quarantine && (hw->config.dif_mode == OCS_HW_DIF_MODE_SEPARATE) && 4085 (iparam->fcp_tgt.dif_oper != OCS_HW_DIF_OPER_DISABLED)) { 4086 io->quarantine = TRUE; 4087 } 4088 4089 ocs_hw_io_ini_sge(hw, io, iparam->fcp_ini.cmnd, iparam->fcp_ini.cmnd_size, 4090 iparam->fcp_ini.rsp); 4091 4092 if (sli_fcp_iread64_wqe(&hw->sli, io->wqe.wqebuf, hw->sli.config.wqe_size, &io->def_sgl, io->first_data_sge, len, 4093 io->indicator, io->reqtag, SLI4_CQ_DEFAULT, rpi, rnode, 4094 iparam->fcp_ini.dif_oper, iparam->fcp_ini.blk_size, 4095 iparam->fcp_ini.timeout)) { 4096 ocs_log_err(hw->os, "IREAD WQE error\n"); 4097 rc = OCS_HW_RTN_ERROR; 4098 } 4099 break; 4100 case OCS_HW_IO_INITIATOR_WRITE: 4101 ocs_hw_io_ini_sge(hw, io, iparam->fcp_ini.cmnd, iparam->fcp_ini.cmnd_size, 4102 iparam->fcp_ini.rsp); 4103 4104 if (sli_fcp_iwrite64_wqe(&hw->sli, io->wqe.wqebuf, hw->sli.config.wqe_size, &io->def_sgl, io->first_data_sge, 4105 len, iparam->fcp_ini.first_burst, 4106 io->indicator, io->reqtag, 4107 SLI4_CQ_DEFAULT, rpi, rnode, 4108 iparam->fcp_ini.dif_oper, iparam->fcp_ini.blk_size, 4109 iparam->fcp_ini.timeout)) { 4110 ocs_log_err(hw->os, "IWRITE WQE error\n"); 4111 rc = OCS_HW_RTN_ERROR; 4112 } 4113 break; 4114 case OCS_HW_IO_INITIATOR_NODATA: 4115 ocs_hw_io_ini_sge(hw, io, iparam->fcp_ini.cmnd, iparam->fcp_ini.cmnd_size, 4116 iparam->fcp_ini.rsp); 4117 4118 if (sli_fcp_icmnd64_wqe(&hw->sli, io->wqe.wqebuf, hw->sli.config.wqe_size, &io->def_sgl, 4119 io->indicator, io->reqtag, SLI4_CQ_DEFAULT, 4120 rpi, rnode, iparam->fcp_ini.timeout)) { 4121 ocs_log_err(hw->os, "ICMND WQE error\n"); 4122 rc = OCS_HW_RTN_ERROR; 4123 } 4124 break; 4125 case OCS_HW_IO_TARGET_WRITE: { 4126 uint16_t flags = iparam->fcp_tgt.flags; 4127 fcp_xfer_rdy_iu_t *xfer = io->xfer_rdy.virt; 4128 4129 /* 4130 * Fill in the XFER_RDY for IF_TYPE 0 devices 4131 */ 4132 *((uint32_t *)xfer->fcp_data_ro) = ocs_htobe32(iparam->fcp_tgt.offset); 4133 *((uint32_t *)xfer->fcp_burst_len) = ocs_htobe32(len); 4134 *((uint32_t *)xfer->rsvd) = 0; 4135 4136 if (io->xbusy) { 4137 flags |= SLI4_IO_CONTINUATION; 4138 } else { 4139 flags &= ~SLI4_IO_CONTINUATION; 4140 } 4141 4142 io->tgt_wqe_timeout = iparam->fcp_tgt.timeout; 4143 4144 /* 4145 * If use_dif_quarantine workaround is in effect, and this is a DIF enabled IO 4146 * then mark the target write IO for quarantine 4147 */ 4148 if (hw->workaround.use_dif_quarantine && (hw->config.dif_mode == OCS_HW_DIF_MODE_SEPARATE) && 4149 (iparam->fcp_tgt.dif_oper != OCS_HW_DIF_OPER_DISABLED)) { 4150 io->quarantine = TRUE; 4151 } 4152 4153 /* 4154 * BZ 161832 Workaround: 4155 * Check for use_dif_sec_xri workaround. Note, even though the first dataphase 4156 * doesn't really need a secondary XRI, we allocate one anyway, as this avoids the 4157 * potential for deadlock where all XRI's are allocated as primaries to IOs that 4158 * are on hw->sec_hio_wait_list. If this secondary XRI is not for the first 4159 * data phase, it is marked for quarantine. 4160 */ 4161 if (hw->workaround.use_dif_sec_xri && (iparam->fcp_tgt.dif_oper != OCS_HW_DIF_OPER_DISABLED)) { 4162 /* 4163 * If we have allocated a chained SGL for skyhawk, then 4164 * we can re-use this for the sec_hio. 4165 */ 4166 if (io->ovfl_io != NULL) { 4167 io->sec_hio = io->ovfl_io; 4168 io->sec_hio->quarantine = TRUE; 4169 } else { 4170 io->sec_hio = ocs_hw_io_alloc(hw); 4171 } 4172 if (io->sec_hio == NULL) { 4173 /* Failed to allocate, so save full request context and put 4174 * this IO on the wait list 4175 */ 4176 io->sec_iparam = *iparam; 4177 io->sec_len = len; 4178 ocs_lock(&hw->io_lock); 4179 ocs_list_remove(&hw->io_inuse, io); 4180 ocs_list_add_tail(&hw->sec_hio_wait_list, io); 4181 io->state = OCS_HW_IO_STATE_WAIT_SEC_HIO; 4182 hw->sec_hio_wait_count++; 4183 ocs_unlock(&hw->io_lock); 4184 send_wqe = FALSE; 4185 /* Done */ 4186 break; 4187 } 4188 /* We quarantine the secondary IO if this is the second or subsequent data phase */ 4189 if (io->xbusy) { 4190 io->sec_hio->quarantine = TRUE; 4191 } 4192 } 4193 4194 /* 4195 * If not the first data phase, and io->sec_hio has been allocated, then issue 4196 * FCP_CONT_TRECEIVE64 WQE, otherwise use the usual FCP_TRECEIVE64 WQE 4197 */ 4198 if (io->xbusy && (io->sec_hio != NULL)) { 4199 if (sli_fcp_cont_treceive64_wqe(&hw->sli, io->wqe.wqebuf, hw->sli.config.wqe_size, &io->def_sgl, io->first_data_sge, 4200 iparam->fcp_tgt.offset, len, io->indicator, io->sec_hio->indicator, 4201 io->reqtag, SLI4_CQ_DEFAULT, 4202 iparam->fcp_tgt.ox_id, rpi, rnode, 4203 flags, 4204 iparam->fcp_tgt.dif_oper, iparam->fcp_tgt.blk_size, 4205 iparam->fcp_tgt.cs_ctl, iparam->fcp_tgt.app_id)) { 4206 ocs_log_err(hw->os, "TRECEIVE WQE error\n"); 4207 rc = OCS_HW_RTN_ERROR; 4208 } 4209 } else { 4210 if (sli_fcp_treceive64_wqe(&hw->sli, io->wqe.wqebuf, hw->sli.config.wqe_size, &io->def_sgl, io->first_data_sge, 4211 iparam->fcp_tgt.offset, len, io->indicator, io->reqtag, 4212 SLI4_CQ_DEFAULT, 4213 iparam->fcp_tgt.ox_id, rpi, rnode, 4214 flags, 4215 iparam->fcp_tgt.dif_oper, iparam->fcp_tgt.blk_size, 4216 iparam->fcp_tgt.cs_ctl, iparam->fcp_tgt.app_id)) { 4217 ocs_log_err(hw->os, "TRECEIVE WQE error\n"); 4218 rc = OCS_HW_RTN_ERROR; 4219 } 4220 } 4221 break; 4222 } 4223 case OCS_HW_IO_TARGET_READ: { 4224 uint16_t flags = iparam->fcp_tgt.flags; 4225 4226 if (io->xbusy) { 4227 flags |= SLI4_IO_CONTINUATION; 4228 } else { 4229 flags &= ~SLI4_IO_CONTINUATION; 4230 } 4231 4232 io->tgt_wqe_timeout = iparam->fcp_tgt.timeout; 4233 if (sli_fcp_tsend64_wqe(&hw->sli, io->wqe.wqebuf, hw->sli.config.wqe_size, &io->def_sgl, io->first_data_sge, 4234 iparam->fcp_tgt.offset, len, io->indicator, io->reqtag, 4235 SLI4_CQ_DEFAULT, 4236 iparam->fcp_tgt.ox_id, rpi, rnode, 4237 flags, 4238 iparam->fcp_tgt.dif_oper, 4239 iparam->fcp_tgt.blk_size, 4240 iparam->fcp_tgt.cs_ctl, 4241 iparam->fcp_tgt.app_id)) { 4242 ocs_log_err(hw->os, "TSEND WQE error\n"); 4243 rc = OCS_HW_RTN_ERROR; 4244 } else if (hw->workaround.retain_tsend_io_length) { 4245 io->length = len; 4246 } 4247 break; 4248 } 4249 case OCS_HW_IO_TARGET_RSP: { 4250 uint16_t flags = iparam->fcp_tgt.flags; 4251 4252 if (io->xbusy) { 4253 flags |= SLI4_IO_CONTINUATION; 4254 } else { 4255 flags &= ~SLI4_IO_CONTINUATION; 4256 } 4257 4258 /* post a new auto xfer ready buffer */ 4259 if (hw->auto_xfer_rdy_enabled && io->is_port_owned) { 4260 if ((io->auto_xfer_rdy_dnrx = ocs_hw_rqpair_auto_xfer_rdy_buffer_post(hw, io, 1))) { 4261 flags |= SLI4_IO_DNRX; 4262 } 4263 } 4264 4265 io->tgt_wqe_timeout = iparam->fcp_tgt.timeout; 4266 if (sli_fcp_trsp64_wqe(&hw->sli, io->wqe.wqebuf, hw->sli.config.wqe_size, 4267 &io->def_sgl, 4268 len, 4269 io->indicator, io->reqtag, 4270 SLI4_CQ_DEFAULT, 4271 iparam->fcp_tgt.ox_id, 4272 rpi, rnode, 4273 flags, iparam->fcp_tgt.cs_ctl, 4274 io->is_port_owned, 4275 iparam->fcp_tgt.app_id)) { 4276 ocs_log_err(hw->os, "TRSP WQE error\n"); 4277 rc = OCS_HW_RTN_ERROR; 4278 } 4279 4280 break; 4281 } 4282 default: 4283 ocs_log_err(hw->os, "unsupported IO type %#x\n", type); 4284 rc = OCS_HW_RTN_ERROR; 4285 } 4286 4287 if (send_wqe && (OCS_HW_RTN_SUCCESS == rc)) { 4288 if (io->wq == NULL) { 4289 io->wq = ocs_hw_queue_next_wq(hw, io); 4290 ocs_hw_assert(io->wq != NULL); 4291 } 4292 4293 io->xbusy = TRUE; 4294 4295 /* 4296 * Add IO to active io wqe list before submitting, in case the 4297 * wcqe processing preempts this thread. 4298 */ 4299 OCS_STAT(hw->tcmd_wq_submit[io->wq->instance]++); 4300 OCS_STAT(io->wq->use_count++); 4301 ocs_hw_add_io_timed_wqe(hw, io); 4302 rc = hw_wq_write(io->wq, &io->wqe); 4303 if (rc >= 0) { 4304 /* non-negative return is success */ 4305 rc = 0; 4306 } else { 4307 /* failed to write wqe, remove from active wqe list */ 4308 ocs_log_err(hw->os, "sli_queue_write failed: %d\n", rc); 4309 io->xbusy = FALSE; 4310 ocs_hw_remove_io_timed_wqe(hw, io); 4311 } 4312 } 4313 4314 return rc; 4315 } 4316 4317 /** 4318 * @brief Send a raw frame 4319 * 4320 * @par Description 4321 * Using the SEND_FRAME_WQE, a frame consisting of header and payload is sent. 4322 * 4323 * @param hw Pointer to HW object. 4324 * @param hdr Pointer to a little endian formatted FC header. 4325 * @param sof Value to use as the frame SOF. 4326 * @param eof Value to use as the frame EOF. 4327 * @param payload Pointer to payload DMA buffer. 4328 * @param ctx Pointer to caller provided send frame context. 4329 * @param callback Callback function. 4330 * @param arg Callback function argument. 4331 * 4332 * @return Returns 0 on success, or a negative error code value on failure. 4333 */ 4334 ocs_hw_rtn_e 4335 ocs_hw_send_frame(ocs_hw_t *hw, fc_header_le_t *hdr, uint8_t sof, uint8_t eof, ocs_dma_t *payload, 4336 ocs_hw_send_frame_context_t *ctx, void (*callback)(void *arg, uint8_t *cqe, int32_t status), void *arg) 4337 { 4338 int32_t rc; 4339 ocs_hw_wqe_t *wqe; 4340 uint32_t xri; 4341 hw_wq_t *wq; 4342 4343 wqe = &ctx->wqe; 4344 4345 /* populate the callback object */ 4346 ctx->hw = hw; 4347 4348 /* Fetch and populate request tag */ 4349 ctx->wqcb = ocs_hw_reqtag_alloc(hw, callback, arg); 4350 if (ctx->wqcb == NULL) { 4351 ocs_log_err(hw->os, "can't allocate request tag\n"); 4352 return OCS_HW_RTN_NO_RESOURCES; 4353 } 4354 4355 /* Choose a work queue, first look for a class[1] wq, otherwise just use wq[0] */ 4356 wq = ocs_varray_iter_next(hw->wq_class_array[1]); 4357 if (wq == NULL) { 4358 wq = hw->hw_wq[0]; 4359 } 4360 4361 /* Set XRI and RX_ID in the header based on which WQ, and which send_frame_io we are using */ 4362 xri = wq->send_frame_io->indicator; 4363 4364 /* Build the send frame WQE */ 4365 rc = sli_send_frame_wqe(&hw->sli, wqe->wqebuf, hw->sli.config.wqe_size, sof, eof, (uint32_t*) hdr, payload, 4366 payload->len, OCS_HW_SEND_FRAME_TIMEOUT, xri, ctx->wqcb->instance_index); 4367 if (rc) { 4368 ocs_log_err(hw->os, "sli_send_frame_wqe failed: %d\n", rc); 4369 return OCS_HW_RTN_ERROR; 4370 } 4371 4372 /* Write to WQ */ 4373 rc = hw_wq_write(wq, wqe); 4374 if (rc) { 4375 ocs_log_err(hw->os, "hw_wq_write failed: %d\n", rc); 4376 return OCS_HW_RTN_ERROR; 4377 } 4378 4379 OCS_STAT(wq->use_count++); 4380 4381 return OCS_HW_RTN_SUCCESS; 4382 } 4383 4384 ocs_hw_rtn_e 4385 ocs_hw_io_register_sgl(ocs_hw_t *hw, ocs_hw_io_t *io, ocs_dma_t *sgl, uint32_t sgl_count) 4386 { 4387 if (sli_get_sgl_preregister(&hw->sli)) { 4388 ocs_log_err(hw->os, "can't use temporary SGL with pre-registered SGLs\n"); 4389 return OCS_HW_RTN_ERROR; 4390 } 4391 io->ovfl_sgl = sgl; 4392 io->ovfl_sgl_count = sgl_count; 4393 io->ovfl_io = NULL; 4394 4395 return OCS_HW_RTN_SUCCESS; 4396 } 4397 4398 static void 4399 ocs_hw_io_restore_sgl(ocs_hw_t *hw, ocs_hw_io_t *io) 4400 { 4401 /* Restore the default */ 4402 io->sgl = &io->def_sgl; 4403 io->sgl_count = io->def_sgl_count; 4404 4405 /* 4406 * For skyhawk, we need to free the IO allocated for the chained 4407 * SGL. For all devices, clear the overflow fields on the IO. 4408 * 4409 * Note: For DIF IOs, we may be using the same XRI for the sec_hio and 4410 * the chained SGLs. If so, then we clear the ovfl_io field 4411 * when the sec_hio is freed. 4412 */ 4413 if (io->ovfl_io != NULL) { 4414 ocs_hw_io_free(hw, io->ovfl_io); 4415 io->ovfl_io = NULL; 4416 } 4417 4418 /* Clear the overflow SGL */ 4419 io->ovfl_sgl = NULL; 4420 io->ovfl_sgl_count = 0; 4421 io->ovfl_lsp = NULL; 4422 } 4423 4424 /** 4425 * @ingroup io 4426 * @brief Initialize the scatter gather list entries of an IO. 4427 * 4428 * @param hw Hardware context. 4429 * @param io Previously-allocated HW IO object. 4430 * @param type Type of IO (target read, target response, and so on). 4431 * 4432 * @return Returns 0 on success, or a non-zero value on failure. 4433 */ 4434 ocs_hw_rtn_e 4435 ocs_hw_io_init_sges(ocs_hw_t *hw, ocs_hw_io_t *io, ocs_hw_io_type_e type) 4436 { 4437 sli4_sge_t *data = NULL; 4438 uint32_t i = 0; 4439 uint32_t skips = 0; 4440 4441 if (!hw || !io) { 4442 ocs_log_err(hw ? hw->os : NULL, "bad parameter hw=%p io=%p\n", 4443 hw, io); 4444 return OCS_HW_RTN_ERROR; 4445 } 4446 4447 /* Clear / reset the scatter-gather list */ 4448 io->sgl = &io->def_sgl; 4449 io->sgl_count = io->def_sgl_count; 4450 io->first_data_sge = 0; 4451 4452 ocs_memset(io->sgl->virt, 0, 2 * sizeof(sli4_sge_t)); 4453 io->n_sge = 0; 4454 io->sge_offset = 0; 4455 4456 io->type = type; 4457 4458 data = io->sgl->virt; 4459 4460 /* 4461 * Some IO types have underlying hardware requirements on the order 4462 * of SGEs. Process all special entries here. 4463 */ 4464 switch (type) { 4465 case OCS_HW_IO_INITIATOR_READ: 4466 case OCS_HW_IO_INITIATOR_WRITE: 4467 case OCS_HW_IO_INITIATOR_NODATA: 4468 /* 4469 * No skips, 2 special for initiator I/Os 4470 * The addresses and length are written later 4471 */ 4472 /* setup command pointer */ 4473 data->sge_type = SLI4_SGE_TYPE_DATA; 4474 data++; 4475 4476 /* setup response pointer */ 4477 data->sge_type = SLI4_SGE_TYPE_DATA; 4478 4479 if (OCS_HW_IO_INITIATOR_NODATA == type) { 4480 data->last = TRUE; 4481 } 4482 data++; 4483 4484 io->n_sge = 2; 4485 break; 4486 case OCS_HW_IO_TARGET_WRITE: 4487 #define OCS_TARGET_WRITE_SKIPS 2 4488 skips = OCS_TARGET_WRITE_SKIPS; 4489 4490 /* populate host resident XFER_RDY buffer */ 4491 data->sge_type = SLI4_SGE_TYPE_DATA; 4492 data->buffer_address_high = ocs_addr32_hi(io->xfer_rdy.phys); 4493 data->buffer_address_low = ocs_addr32_lo(io->xfer_rdy.phys); 4494 data->buffer_length = io->xfer_rdy.size; 4495 data++; 4496 4497 skips--; 4498 4499 io->n_sge = 1; 4500 break; 4501 case OCS_HW_IO_TARGET_READ: 4502 /* 4503 * For FCP_TSEND64, the first 2 entries are SKIP SGE's 4504 */ 4505 #define OCS_TARGET_READ_SKIPS 2 4506 skips = OCS_TARGET_READ_SKIPS; 4507 break; 4508 case OCS_HW_IO_TARGET_RSP: 4509 /* 4510 * No skips, etc. for FCP_TRSP64 4511 */ 4512 break; 4513 default: 4514 ocs_log_err(hw->os, "unsupported IO type %#x\n", type); 4515 return OCS_HW_RTN_ERROR; 4516 } 4517 4518 /* 4519 * Write skip entries 4520 */ 4521 for (i = 0; i < skips; i++) { 4522 data->sge_type = SLI4_SGE_TYPE_SKIP; 4523 data++; 4524 } 4525 4526 io->n_sge += skips; 4527 4528 /* 4529 * Set last 4530 */ 4531 data->last = TRUE; 4532 4533 return OCS_HW_RTN_SUCCESS; 4534 } 4535 4536 /** 4537 * @ingroup io 4538 * @brief Add a T10 PI seed scatter gather list entry. 4539 * 4540 * @param hw Hardware context. 4541 * @param io Previously-allocated HW IO object. 4542 * @param dif_info Pointer to T10 DIF fields, or NULL if no DIF. 4543 * 4544 * @return Returns 0 on success, or a non-zero value on failure. 4545 */ 4546 ocs_hw_rtn_e 4547 ocs_hw_io_add_seed_sge(ocs_hw_t *hw, ocs_hw_io_t *io, ocs_hw_dif_info_t *dif_info) 4548 { 4549 sli4_sge_t *data = NULL; 4550 sli4_diseed_sge_t *dif_seed; 4551 4552 /* If no dif_info, or dif_oper is disabled, then just return success */ 4553 if ((dif_info == NULL) || (dif_info->dif_oper == OCS_HW_DIF_OPER_DISABLED)) { 4554 return OCS_HW_RTN_SUCCESS; 4555 } 4556 4557 if (!hw || !io) { 4558 ocs_log_err(hw ? hw->os : NULL, "bad parameter hw=%p io=%p dif_info=%p\n", 4559 hw, io, dif_info); 4560 return OCS_HW_RTN_ERROR; 4561 } 4562 4563 data = io->sgl->virt; 4564 data += io->n_sge; 4565 4566 /* If we are doing T10 DIF add the DIF Seed SGE */ 4567 ocs_memset(data, 0, sizeof(sli4_diseed_sge_t)); 4568 dif_seed = (sli4_diseed_sge_t *)data; 4569 dif_seed->ref_tag_cmp = dif_info->ref_tag_cmp; 4570 dif_seed->ref_tag_repl = dif_info->ref_tag_repl; 4571 dif_seed->app_tag_repl = dif_info->app_tag_repl; 4572 dif_seed->repl_app_tag = dif_info->repl_app_tag; 4573 if (SLI4_IF_TYPE_LANCER_FC_ETH != hw->sli.if_type) { 4574 dif_seed->atrt = dif_info->disable_app_ref_ffff; 4575 dif_seed->at = dif_info->disable_app_ffff; 4576 } 4577 dif_seed->sge_type = SLI4_SGE_TYPE_DISEED; 4578 /* Workaround for SKH (BZ157233) */ 4579 if (((io->type == OCS_HW_IO_TARGET_WRITE) || (io->type == OCS_HW_IO_INITIATOR_READ)) && 4580 (SLI4_IF_TYPE_LANCER_FC_ETH != hw->sli.if_type) && dif_info->dif_separate) { 4581 dif_seed->sge_type = SLI4_SGE_TYPE_SKIP; 4582 } 4583 4584 dif_seed->app_tag_cmp = dif_info->app_tag_cmp; 4585 dif_seed->dif_blk_size = dif_info->blk_size; 4586 dif_seed->auto_incr_ref_tag = dif_info->auto_incr_ref_tag; 4587 dif_seed->check_app_tag = dif_info->check_app_tag; 4588 dif_seed->check_ref_tag = dif_info->check_ref_tag; 4589 dif_seed->check_crc = dif_info->check_guard; 4590 dif_seed->new_ref_tag = dif_info->repl_ref_tag; 4591 4592 switch(dif_info->dif_oper) { 4593 case OCS_HW_SGE_DIF_OP_IN_NODIF_OUT_CRC: 4594 dif_seed->dif_op_rx = SLI4_SGE_DIF_OP_IN_NODIF_OUT_CRC; 4595 dif_seed->dif_op_tx = SLI4_SGE_DIF_OP_IN_NODIF_OUT_CRC; 4596 break; 4597 case OCS_HW_SGE_DIF_OP_IN_CRC_OUT_NODIF: 4598 dif_seed->dif_op_rx = SLI4_SGE_DIF_OP_IN_CRC_OUT_NODIF; 4599 dif_seed->dif_op_tx = SLI4_SGE_DIF_OP_IN_CRC_OUT_NODIF; 4600 break; 4601 case OCS_HW_SGE_DIF_OP_IN_NODIF_OUT_CHKSUM: 4602 dif_seed->dif_op_rx = SLI4_SGE_DIF_OP_IN_NODIF_OUT_CHKSUM; 4603 dif_seed->dif_op_tx = SLI4_SGE_DIF_OP_IN_NODIF_OUT_CHKSUM; 4604 break; 4605 case OCS_HW_SGE_DIF_OP_IN_CHKSUM_OUT_NODIF: 4606 dif_seed->dif_op_rx = SLI4_SGE_DIF_OP_IN_CHKSUM_OUT_NODIF; 4607 dif_seed->dif_op_tx = SLI4_SGE_DIF_OP_IN_CHKSUM_OUT_NODIF; 4608 break; 4609 case OCS_HW_SGE_DIF_OP_IN_CRC_OUT_CRC: 4610 dif_seed->dif_op_rx = SLI4_SGE_DIF_OP_IN_CRC_OUT_CRC; 4611 dif_seed->dif_op_tx = SLI4_SGE_DIF_OP_IN_CRC_OUT_CRC; 4612 break; 4613 case OCS_HW_SGE_DIF_OP_IN_CHKSUM_OUT_CHKSUM: 4614 dif_seed->dif_op_rx = SLI4_SGE_DIF_OP_IN_CHKSUM_OUT_CHKSUM; 4615 dif_seed->dif_op_tx = SLI4_SGE_DIF_OP_IN_CHKSUM_OUT_CHKSUM; 4616 break; 4617 case OCS_HW_SGE_DIF_OP_IN_CRC_OUT_CHKSUM: 4618 dif_seed->dif_op_rx = SLI4_SGE_DIF_OP_IN_CRC_OUT_CHKSUM; 4619 dif_seed->dif_op_tx = SLI4_SGE_DIF_OP_IN_CRC_OUT_CHKSUM; 4620 break; 4621 case OCS_HW_SGE_DIF_OP_IN_CHKSUM_OUT_CRC: 4622 dif_seed->dif_op_rx = SLI4_SGE_DIF_OP_IN_CHKSUM_OUT_CRC; 4623 dif_seed->dif_op_tx = SLI4_SGE_DIF_OP_IN_CHKSUM_OUT_CRC; 4624 break; 4625 case OCS_HW_SGE_DIF_OP_IN_RAW_OUT_RAW: 4626 dif_seed->dif_op_rx = SLI4_SGE_DIF_OP_IN_RAW_OUT_RAW; 4627 dif_seed->dif_op_tx = SLI4_SGE_DIF_OP_IN_RAW_OUT_RAW; 4628 break; 4629 default: 4630 ocs_log_err(hw->os, "unsupported DIF operation %#x\n", 4631 dif_info->dif_oper); 4632 return OCS_HW_RTN_ERROR; 4633 } 4634 4635 /* 4636 * Set last, clear previous last 4637 */ 4638 data->last = TRUE; 4639 if (io->n_sge) { 4640 data[-1].last = FALSE; 4641 } 4642 4643 io->n_sge++; 4644 4645 return OCS_HW_RTN_SUCCESS; 4646 } 4647 4648 static ocs_hw_rtn_e 4649 ocs_hw_io_overflow_sgl(ocs_hw_t *hw, ocs_hw_io_t *io) 4650 { 4651 sli4_lsp_sge_t *lsp; 4652 4653 /* fail if we're already pointing to the overflow SGL */ 4654 if (io->sgl == io->ovfl_sgl) { 4655 return OCS_HW_RTN_ERROR; 4656 } 4657 4658 /* 4659 * For skyhawk, we can use another SGL to extend the SGL list. The 4660 * Chained entry must not be in the first 4 entries. 4661 * 4662 * Note: For DIF enabled IOs, we will use the ovfl_io for the sec_hio. 4663 */ 4664 if (sli_get_sgl_preregister(&hw->sli) && 4665 io->def_sgl_count > 4 && 4666 io->ovfl_io == NULL && 4667 ((SLI4_IF_TYPE_BE3_SKH_PF == sli_get_if_type(&hw->sli)) || 4668 (SLI4_IF_TYPE_BE3_SKH_VF == sli_get_if_type(&hw->sli)))) { 4669 io->ovfl_io = ocs_hw_io_alloc(hw); 4670 if (io->ovfl_io != NULL) { 4671 /* 4672 * Note: We can't call ocs_hw_io_register_sgl() here 4673 * because it checks that SGLs are not pre-registered 4674 * and for shyhawk, preregistered SGLs are required. 4675 */ 4676 io->ovfl_sgl = &io->ovfl_io->def_sgl; 4677 io->ovfl_sgl_count = io->ovfl_io->def_sgl_count; 4678 } 4679 } 4680 4681 /* fail if we don't have an overflow SGL registered */ 4682 if (io->ovfl_io == NULL || io->ovfl_sgl == NULL) { 4683 return OCS_HW_RTN_ERROR; 4684 } 4685 4686 /* 4687 * Overflow, we need to put a link SGE in the last location of the current SGL, after 4688 * copying the the last SGE to the overflow SGL 4689 */ 4690 4691 ((sli4_sge_t*)io->ovfl_sgl->virt)[0] = ((sli4_sge_t*)io->sgl->virt)[io->n_sge - 1]; 4692 4693 lsp = &((sli4_lsp_sge_t*)io->sgl->virt)[io->n_sge - 1]; 4694 ocs_memset(lsp, 0, sizeof(*lsp)); 4695 4696 if ((SLI4_IF_TYPE_BE3_SKH_PF == sli_get_if_type(&hw->sli)) || 4697 (SLI4_IF_TYPE_BE3_SKH_VF == sli_get_if_type(&hw->sli))) { 4698 sli_skh_chain_sge_build(&hw->sli, 4699 (sli4_sge_t*)lsp, 4700 io->ovfl_io->indicator, 4701 0, /* frag_num */ 4702 0); /* offset */ 4703 } else { 4704 lsp->buffer_address_high = ocs_addr32_hi(io->ovfl_sgl->phys); 4705 lsp->buffer_address_low = ocs_addr32_lo(io->ovfl_sgl->phys); 4706 lsp->sge_type = SLI4_SGE_TYPE_LSP; 4707 lsp->last = 0; 4708 io->ovfl_lsp = lsp; 4709 io->ovfl_lsp->segment_length = sizeof(sli4_sge_t); 4710 } 4711 4712 /* Update the current SGL pointer, and n_sgl */ 4713 io->sgl = io->ovfl_sgl; 4714 io->sgl_count = io->ovfl_sgl_count; 4715 io->n_sge = 1; 4716 4717 return OCS_HW_RTN_SUCCESS; 4718 } 4719 4720 /** 4721 * @ingroup io 4722 * @brief Add a scatter gather list entry to an IO. 4723 * 4724 * @param hw Hardware context. 4725 * @param io Previously-allocated HW IO object. 4726 * @param addr Physical address. 4727 * @param length Length of memory pointed to by @c addr. 4728 * 4729 * @return Returns 0 on success, or a non-zero value on failure. 4730 */ 4731 ocs_hw_rtn_e 4732 ocs_hw_io_add_sge(ocs_hw_t *hw, ocs_hw_io_t *io, uintptr_t addr, uint32_t length) 4733 { 4734 sli4_sge_t *data = NULL; 4735 4736 if (!hw || !io || !addr || !length) { 4737 ocs_log_err(hw ? hw->os : NULL, 4738 "bad parameter hw=%p io=%p addr=%lx length=%u\n", 4739 hw, io, addr, length); 4740 return OCS_HW_RTN_ERROR; 4741 } 4742 4743 if ((length != 0) && (io->n_sge + 1) > io->sgl_count) { 4744 if (ocs_hw_io_overflow_sgl(hw, io) != OCS_HW_RTN_SUCCESS) { 4745 ocs_log_err(hw->os, "SGL full (%d)\n", io->n_sge); 4746 return OCS_HW_RTN_ERROR; 4747 } 4748 } 4749 4750 if (length > sli_get_max_sge(&hw->sli)) { 4751 ocs_log_err(hw->os, "length of SGE %d bigger than allowed %d\n", 4752 length, sli_get_max_sge(&hw->sli)); 4753 return OCS_HW_RTN_ERROR; 4754 } 4755 4756 data = io->sgl->virt; 4757 data += io->n_sge; 4758 4759 data->sge_type = SLI4_SGE_TYPE_DATA; 4760 data->buffer_address_high = ocs_addr32_hi(addr); 4761 data->buffer_address_low = ocs_addr32_lo(addr); 4762 data->buffer_length = length; 4763 data->data_offset = io->sge_offset; 4764 /* 4765 * Always assume this is the last entry and mark as such. 4766 * If this is not the first entry unset the "last SGE" 4767 * indication for the previous entry 4768 */ 4769 data->last = TRUE; 4770 if (io->n_sge) { 4771 data[-1].last = FALSE; 4772 } 4773 4774 /* Set first_data_bde if not previously set */ 4775 if (io->first_data_sge == 0) { 4776 io->first_data_sge = io->n_sge; 4777 } 4778 4779 io->sge_offset += length; 4780 io->n_sge++; 4781 4782 /* Update the linked segment length (only executed after overflow has begun) */ 4783 if (io->ovfl_lsp != NULL) { 4784 io->ovfl_lsp->segment_length = io->n_sge * sizeof(sli4_sge_t); 4785 } 4786 4787 return OCS_HW_RTN_SUCCESS; 4788 } 4789 4790 /** 4791 * @ingroup io 4792 * @brief Add a T10 DIF scatter gather list entry to an IO. 4793 * 4794 * @param hw Hardware context. 4795 * @param io Previously-allocated HW IO object. 4796 * @param addr DIF physical address. 4797 * 4798 * @return Returns 0 on success, or a non-zero value on failure. 4799 */ 4800 ocs_hw_rtn_e 4801 ocs_hw_io_add_dif_sge(ocs_hw_t *hw, ocs_hw_io_t *io, uintptr_t addr) 4802 { 4803 sli4_dif_sge_t *data = NULL; 4804 4805 if (!hw || !io || !addr) { 4806 ocs_log_err(hw ? hw->os : NULL, 4807 "bad parameter hw=%p io=%p addr=%lx\n", 4808 hw, io, addr); 4809 return OCS_HW_RTN_ERROR; 4810 } 4811 4812 if ((io->n_sge + 1) > hw->config.n_sgl) { 4813 if (ocs_hw_io_overflow_sgl(hw, io) != OCS_HW_RTN_ERROR) { 4814 ocs_log_err(hw->os, "SGL full (%d)\n", io->n_sge); 4815 return OCS_HW_RTN_ERROR; 4816 } 4817 } 4818 4819 data = io->sgl->virt; 4820 data += io->n_sge; 4821 4822 data->sge_type = SLI4_SGE_TYPE_DIF; 4823 /* Workaround for SKH (BZ157233) */ 4824 if (((io->type == OCS_HW_IO_TARGET_WRITE) || (io->type == OCS_HW_IO_INITIATOR_READ)) && 4825 (SLI4_IF_TYPE_LANCER_FC_ETH != hw->sli.if_type)) { 4826 data->sge_type = SLI4_SGE_TYPE_SKIP; 4827 } 4828 4829 data->buffer_address_high = ocs_addr32_hi(addr); 4830 data->buffer_address_low = ocs_addr32_lo(addr); 4831 4832 /* 4833 * Always assume this is the last entry and mark as such. 4834 * If this is not the first entry unset the "last SGE" 4835 * indication for the previous entry 4836 */ 4837 data->last = TRUE; 4838 if (io->n_sge) { 4839 data[-1].last = FALSE; 4840 } 4841 4842 io->n_sge++; 4843 4844 return OCS_HW_RTN_SUCCESS; 4845 } 4846 4847 /** 4848 * @ingroup io 4849 * @brief Abort a previously-started IO. 4850 * 4851 * @param hw Hardware context. 4852 * @param io_to_abort The IO to abort. 4853 * @param send_abts Boolean to have the hardware automatically 4854 * generate an ABTS. 4855 * @param cb Function call upon completion of the abort (may be NULL). 4856 * @param arg Argument to pass to abort completion function. 4857 * 4858 * @return Returns 0 on success, or a non-zero value on failure. 4859 */ 4860 ocs_hw_rtn_e 4861 ocs_hw_io_abort(ocs_hw_t *hw, ocs_hw_io_t *io_to_abort, uint32_t send_abts, void *cb, void *arg) 4862 { 4863 sli4_abort_type_e atype = SLI_ABORT_MAX; 4864 uint32_t id = 0, mask = 0; 4865 ocs_hw_rtn_e rc = OCS_HW_RTN_SUCCESS; 4866 hw_wq_callback_t *wqcb; 4867 4868 if (!hw || !io_to_abort) { 4869 ocs_log_err(hw ? hw->os : NULL, 4870 "bad parameter hw=%p io=%p\n", 4871 hw, io_to_abort); 4872 return OCS_HW_RTN_ERROR; 4873 } 4874 4875 if (hw->state != OCS_HW_STATE_ACTIVE) { 4876 ocs_log_err(hw->os, "cannot send IO abort, HW state=%d\n", 4877 hw->state); 4878 return OCS_HW_RTN_ERROR; 4879 } 4880 4881 /* take a reference on IO being aborted */ 4882 if (ocs_ref_get_unless_zero(&io_to_abort->ref) == 0) { 4883 /* command no longer active */ 4884 ocs_log_test(hw ? hw->os : NULL, 4885 "io not active xri=0x%x tag=0x%x\n", 4886 io_to_abort->indicator, io_to_abort->reqtag); 4887 return OCS_HW_RTN_IO_NOT_ACTIVE; 4888 } 4889 4890 /* non-port owned XRI checks */ 4891 /* Must have a valid WQ reference */ 4892 if (io_to_abort->wq == NULL) { 4893 ocs_log_test(hw->os, "io_to_abort xri=0x%x not active on WQ\n", 4894 io_to_abort->indicator); 4895 ocs_ref_put(&io_to_abort->ref); /* ocs_ref_get(): same function */ 4896 return OCS_HW_RTN_IO_NOT_ACTIVE; 4897 } 4898 4899 /* Validation checks complete; now check to see if already being aborted */ 4900 ocs_lock(&hw->io_abort_lock); 4901 if (io_to_abort->abort_in_progress) { 4902 ocs_unlock(&hw->io_abort_lock); 4903 ocs_ref_put(&io_to_abort->ref); /* ocs_ref_get(): same function */ 4904 ocs_log_debug(hw ? hw->os : NULL, 4905 "io already being aborted xri=0x%x tag=0x%x\n", 4906 io_to_abort->indicator, io_to_abort->reqtag); 4907 return OCS_HW_RTN_IO_ABORT_IN_PROGRESS; 4908 } 4909 4910 /* 4911 * This IO is not already being aborted. Set flag so we won't try to 4912 * abort it again. After all, we only have one abort_done callback. 4913 */ 4914 io_to_abort->abort_in_progress = 1; 4915 ocs_unlock(&hw->io_abort_lock); 4916 4917 /* 4918 * If we got here, the possibilities are: 4919 * - host owned xri 4920 * - io_to_abort->wq_index != UINT32_MAX 4921 * - submit ABORT_WQE to same WQ 4922 * - port owned xri: 4923 * - rxri: io_to_abort->wq_index == UINT32_MAX 4924 * - submit ABORT_WQE to any WQ 4925 * - non-rxri 4926 * - io_to_abort->index != UINT32_MAX 4927 * - submit ABORT_WQE to same WQ 4928 * - io_to_abort->index == UINT32_MAX 4929 * - submit ABORT_WQE to any WQ 4930 */ 4931 io_to_abort->abort_done = cb; 4932 io_to_abort->abort_arg = arg; 4933 4934 atype = SLI_ABORT_XRI; 4935 id = io_to_abort->indicator; 4936 4937 /* Allocate a request tag for the abort portion of this IO */ 4938 wqcb = ocs_hw_reqtag_alloc(hw, ocs_hw_wq_process_abort, io_to_abort); 4939 if (wqcb == NULL) { 4940 ocs_log_err(hw->os, "can't allocate request tag\n"); 4941 return OCS_HW_RTN_NO_RESOURCES; 4942 } 4943 io_to_abort->abort_reqtag = wqcb->instance_index; 4944 4945 /* 4946 * If the wqe is on the pending list, then set this wqe to be 4947 * aborted when the IO's wqe is removed from the list. 4948 */ 4949 if (io_to_abort->wq != NULL) { 4950 sli_queue_lock(io_to_abort->wq->queue); 4951 if (ocs_list_on_list(&io_to_abort->wqe.link)) { 4952 io_to_abort->wqe.abort_wqe_submit_needed = 1; 4953 io_to_abort->wqe.send_abts = send_abts; 4954 io_to_abort->wqe.id = id; 4955 io_to_abort->wqe.abort_reqtag = io_to_abort->abort_reqtag; 4956 sli_queue_unlock(io_to_abort->wq->queue); 4957 return 0; 4958 } 4959 sli_queue_unlock(io_to_abort->wq->queue); 4960 } 4961 4962 if (sli_abort_wqe(&hw->sli, io_to_abort->wqe.wqebuf, hw->sli.config.wqe_size, atype, send_abts, id, mask, 4963 io_to_abort->abort_reqtag, SLI4_CQ_DEFAULT)) { 4964 ocs_log_err(hw->os, "ABORT WQE error\n"); 4965 io_to_abort->abort_reqtag = UINT32_MAX; 4966 ocs_hw_reqtag_free(hw, wqcb); 4967 rc = OCS_HW_RTN_ERROR; 4968 } 4969 4970 if (OCS_HW_RTN_SUCCESS == rc) { 4971 if (io_to_abort->wq == NULL) { 4972 io_to_abort->wq = ocs_hw_queue_next_wq(hw, io_to_abort); 4973 ocs_hw_assert(io_to_abort->wq != NULL); 4974 } 4975 /* ABORT_WQE does not actually utilize an XRI on the Port, 4976 * therefore, keep xbusy as-is to track the exchange's state, 4977 * not the ABORT_WQE's state 4978 */ 4979 rc = hw_wq_write(io_to_abort->wq, &io_to_abort->wqe); 4980 if (rc > 0) { 4981 /* non-negative return is success */ 4982 rc = 0; 4983 /* can't abort an abort so skip adding to timed wqe list */ 4984 } 4985 } 4986 4987 if (OCS_HW_RTN_SUCCESS != rc) { 4988 ocs_lock(&hw->io_abort_lock); 4989 io_to_abort->abort_in_progress = 0; 4990 ocs_unlock(&hw->io_abort_lock); 4991 ocs_ref_put(&io_to_abort->ref); /* ocs_ref_get(): same function */ 4992 } 4993 return rc; 4994 } 4995 4996 /** 4997 * @ingroup io 4998 * @brief Return the OX_ID/RX_ID of the IO. 4999 * 5000 * @param hw Hardware context. 5001 * @param io HW IO object. 5002 * 5003 * @return Returns X_ID on success, or -1 on failure. 5004 */ 5005 int32_t 5006 ocs_hw_io_get_xid(ocs_hw_t *hw, ocs_hw_io_t *io) 5007 { 5008 if (!hw || !io) { 5009 ocs_log_err(hw ? hw->os : NULL, 5010 "bad parameter hw=%p io=%p\n", hw, io); 5011 return -1; 5012 } 5013 5014 return io->indicator; 5015 } 5016 5017 typedef struct ocs_hw_fw_write_cb_arg { 5018 ocs_hw_fw_cb_t cb; 5019 void *arg; 5020 } ocs_hw_fw_write_cb_arg_t; 5021 5022 typedef struct ocs_hw_sfp_cb_arg { 5023 ocs_hw_sfp_cb_t cb; 5024 void *arg; 5025 ocs_dma_t payload; 5026 } ocs_hw_sfp_cb_arg_t; 5027 5028 typedef struct ocs_hw_temp_cb_arg { 5029 ocs_hw_temp_cb_t cb; 5030 void *arg; 5031 } ocs_hw_temp_cb_arg_t; 5032 5033 typedef struct ocs_hw_link_stat_cb_arg { 5034 ocs_hw_link_stat_cb_t cb; 5035 void *arg; 5036 } ocs_hw_link_stat_cb_arg_t; 5037 5038 typedef struct ocs_hw_host_stat_cb_arg { 5039 ocs_hw_host_stat_cb_t cb; 5040 void *arg; 5041 } ocs_hw_host_stat_cb_arg_t; 5042 5043 typedef struct ocs_hw_dump_get_cb_arg { 5044 ocs_hw_dump_get_cb_t cb; 5045 void *arg; 5046 void *mbox_cmd; 5047 } ocs_hw_dump_get_cb_arg_t; 5048 5049 typedef struct ocs_hw_dump_clear_cb_arg { 5050 ocs_hw_dump_clear_cb_t cb; 5051 void *arg; 5052 void *mbox_cmd; 5053 } ocs_hw_dump_clear_cb_arg_t; 5054 5055 /** 5056 * @brief Write a portion of a firmware image to the device. 5057 * 5058 * @par Description 5059 * Calls the correct firmware write function based on the device type. 5060 * 5061 * @param hw Hardware context. 5062 * @param dma DMA structure containing the firmware image chunk. 5063 * @param size Size of the firmware image chunk. 5064 * @param offset Offset, in bytes, from the beginning of the firmware image. 5065 * @param last True if this is the last chunk of the image. 5066 * Causes the image to be committed to flash. 5067 * @param cb Pointer to a callback function that is called when the command completes. 5068 * The callback function prototype is 5069 * <tt>void cb(int32_t status, uint32_t bytes_written, void *arg)</tt>. 5070 * @param arg Pointer to be passed to the callback function. 5071 * 5072 * @return Returns 0 on success, or a non-zero value on failure. 5073 */ 5074 ocs_hw_rtn_e 5075 ocs_hw_firmware_write(ocs_hw_t *hw, ocs_dma_t *dma, uint32_t size, uint32_t offset, int last, ocs_hw_fw_cb_t cb, void *arg) 5076 { 5077 if (hw->sli.if_type == SLI4_IF_TYPE_LANCER_FC_ETH) { 5078 return ocs_hw_firmware_write_lancer(hw, dma, size, offset, last, cb, arg); 5079 } else { 5080 /* Write firmware_write for BE3/Skyhawk not supported */ 5081 return -1; 5082 } 5083 } 5084 5085 /** 5086 * @brief Write a portion of a firmware image to the Emulex XE201 ASIC (Lancer). 5087 * 5088 * @par Description 5089 * Creates a SLI_CONFIG mailbox command, fills it with the correct values to write a 5090 * firmware image chunk, and then sends the command with ocs_hw_command(). On completion, 5091 * the callback function ocs_hw_fw_write_cb() gets called to free the mailbox 5092 * and to signal the caller that the write has completed. 5093 * 5094 * @param hw Hardware context. 5095 * @param dma DMA structure containing the firmware image chunk. 5096 * @param size Size of the firmware image chunk. 5097 * @param offset Offset, in bytes, from the beginning of the firmware image. 5098 * @param last True if this is the last chunk of the image. Causes the image to be committed to flash. 5099 * @param cb Pointer to a callback function that is called when the command completes. 5100 * The callback function prototype is 5101 * <tt>void cb(int32_t status, uint32_t bytes_written, void *arg)</tt>. 5102 * @param arg Pointer to be passed to the callback function. 5103 * 5104 * @return Returns 0 on success, or a non-zero value on failure. 5105 */ 5106 ocs_hw_rtn_e 5107 ocs_hw_firmware_write_lancer(ocs_hw_t *hw, ocs_dma_t *dma, uint32_t size, uint32_t offset, int last, ocs_hw_fw_cb_t cb, void *arg) 5108 { 5109 ocs_hw_rtn_e rc = OCS_HW_RTN_ERROR; 5110 uint8_t *mbxdata; 5111 ocs_hw_fw_write_cb_arg_t *cb_arg; 5112 int noc=0; /* No Commit bit - set to 1 for testing */ 5113 5114 if (SLI4_IF_TYPE_LANCER_FC_ETH != sli_get_if_type(&hw->sli)) { 5115 ocs_log_test(hw->os, "Function only supported for I/F type 2\n"); 5116 return OCS_HW_RTN_ERROR; 5117 } 5118 5119 mbxdata = ocs_malloc(hw->os, SLI4_BMBX_SIZE, OCS_M_ZERO | OCS_M_NOWAIT); 5120 if (mbxdata == NULL) { 5121 ocs_log_err(hw->os, "failed to malloc mbox\n"); 5122 return OCS_HW_RTN_NO_MEMORY; 5123 } 5124 5125 cb_arg = ocs_malloc(hw->os, sizeof(ocs_hw_fw_write_cb_arg_t), OCS_M_NOWAIT); 5126 if (cb_arg == NULL) { 5127 ocs_log_err(hw->os, "failed to malloc cb_arg\n"); 5128 ocs_free(hw->os, mbxdata, SLI4_BMBX_SIZE); 5129 return OCS_HW_RTN_NO_MEMORY; 5130 } 5131 5132 cb_arg->cb = cb; 5133 cb_arg->arg = arg; 5134 5135 if (sli_cmd_common_write_object(&hw->sli, mbxdata, SLI4_BMBX_SIZE, noc, last, 5136 size, offset, "/prg/", dma)) { 5137 rc = ocs_hw_command(hw, mbxdata, OCS_CMD_NOWAIT, ocs_hw_cb_fw_write, cb_arg); 5138 } 5139 5140 if (rc != OCS_HW_RTN_SUCCESS) { 5141 ocs_log_test(hw->os, "COMMON_WRITE_OBJECT failed\n"); 5142 ocs_free(hw->os, mbxdata, SLI4_BMBX_SIZE); 5143 ocs_free(hw->os, cb_arg, sizeof(ocs_hw_fw_write_cb_arg_t)); 5144 } 5145 5146 return rc; 5147 5148 } 5149 5150 /** 5151 * @brief Called when the WRITE OBJECT command completes. 5152 * 5153 * @par Description 5154 * Get the number of bytes actually written out of the response, free the mailbox 5155 * that was malloc'd by ocs_hw_firmware_write(), 5156 * then call the callback and pass the status and bytes written. 5157 * 5158 * @param hw Hardware context. 5159 * @param status Status field from the mbox completion. 5160 * @param mqe Mailbox response structure. 5161 * @param arg Pointer to a callback function that signals the caller that the command is done. 5162 * The callback function prototype is <tt>void cb(int32_t status, uint32_t bytes_written)</tt>. 5163 * 5164 * @return Returns 0. 5165 */ 5166 static int32_t 5167 ocs_hw_cb_fw_write(ocs_hw_t *hw, int32_t status, uint8_t *mqe, void *arg) 5168 { 5169 5170 sli4_cmd_sli_config_t* mbox_rsp = (sli4_cmd_sli_config_t*) mqe; 5171 sli4_res_common_write_object_t* wr_obj_rsp = (sli4_res_common_write_object_t*) &(mbox_rsp->payload.embed); 5172 ocs_hw_fw_write_cb_arg_t *cb_arg = arg; 5173 uint32_t bytes_written; 5174 uint16_t mbox_status; 5175 uint32_t change_status; 5176 5177 bytes_written = wr_obj_rsp->actual_write_length; 5178 mbox_status = mbox_rsp->hdr.status; 5179 change_status = wr_obj_rsp->change_status; 5180 5181 ocs_free(hw->os, mqe, SLI4_BMBX_SIZE); 5182 5183 if (cb_arg) { 5184 if (cb_arg->cb) { 5185 if ((status == 0) && mbox_status) { 5186 status = mbox_status; 5187 } 5188 cb_arg->cb(status, bytes_written, change_status, cb_arg->arg); 5189 } 5190 5191 ocs_free(hw->os, cb_arg, sizeof(ocs_hw_fw_write_cb_arg_t)); 5192 } 5193 5194 return 0; 5195 5196 } 5197 5198 /** 5199 * @brief Called when the READ_TRANSCEIVER_DATA command completes. 5200 * 5201 * @par Description 5202 * Get the number of bytes read out of the response, free the mailbox that was malloc'd 5203 * by ocs_hw_get_sfp(), then call the callback and pass the status and bytes written. 5204 * 5205 * @param hw Hardware context. 5206 * @param status Status field from the mbox completion. 5207 * @param mqe Mailbox response structure. 5208 * @param arg Pointer to a callback function that signals the caller that the command is done. 5209 * The callback function prototype is 5210 * <tt>void cb(int32_t status, uint32_t bytes_written, uint32_t *data, void *arg)</tt>. 5211 * 5212 * @return Returns 0. 5213 */ 5214 static int32_t 5215 ocs_hw_cb_sfp(ocs_hw_t *hw, int32_t status, uint8_t *mqe, void *arg) 5216 { 5217 5218 ocs_hw_sfp_cb_arg_t *cb_arg = arg; 5219 ocs_dma_t *payload = NULL; 5220 sli4_res_common_read_transceiver_data_t* mbox_rsp = NULL; 5221 uint32_t bytes_written; 5222 5223 if (cb_arg) { 5224 payload = &(cb_arg->payload); 5225 if (cb_arg->cb) { 5226 mbox_rsp = (sli4_res_common_read_transceiver_data_t*) payload->virt; 5227 bytes_written = mbox_rsp->hdr.response_length; 5228 if ((status == 0) && mbox_rsp->hdr.status) { 5229 status = mbox_rsp->hdr.status; 5230 } 5231 cb_arg->cb(hw->os, status, bytes_written, mbox_rsp->page_data, cb_arg->arg); 5232 } 5233 5234 ocs_dma_free(hw->os, &cb_arg->payload); 5235 ocs_free(hw->os, cb_arg, sizeof(ocs_hw_sfp_cb_arg_t)); 5236 } 5237 5238 ocs_free(hw->os, mqe, SLI4_BMBX_SIZE); 5239 return 0; 5240 } 5241 5242 /** 5243 * @ingroup io 5244 * @brief Function to retrieve the SFP information. 5245 * 5246 * @param hw Hardware context. 5247 * @param page The page of SFP data to retrieve (0xa0 or 0xa2). 5248 * @param cb Function call upon completion of sending the data (may be NULL). 5249 * @param arg Argument to pass to IO completion function. 5250 * 5251 * @return Returns OCS_HW_RTN_SUCCESS, OCS_HW_RTN_ERROR, or OCS_HW_RTN_NO_MEMORY. 5252 */ 5253 ocs_hw_rtn_e 5254 ocs_hw_get_sfp(ocs_hw_t *hw, uint16_t page, ocs_hw_sfp_cb_t cb, void *arg) 5255 { 5256 ocs_hw_rtn_e rc = OCS_HW_RTN_ERROR; 5257 ocs_hw_sfp_cb_arg_t *cb_arg; 5258 uint8_t *mbxdata; 5259 5260 /* mbxdata holds the header of the command */ 5261 mbxdata = ocs_malloc(hw->os, SLI4_BMBX_SIZE, OCS_M_ZERO | OCS_M_NOWAIT); 5262 if (mbxdata == NULL) { 5263 ocs_log_err(hw->os, "failed to malloc mbox\n"); 5264 return OCS_HW_RTN_NO_MEMORY; 5265 } 5266 5267 /* cb_arg holds the data that will be passed to the callback on completion */ 5268 cb_arg = ocs_malloc(hw->os, sizeof(ocs_hw_sfp_cb_arg_t), OCS_M_NOWAIT); 5269 if (cb_arg == NULL) { 5270 ocs_log_err(hw->os, "failed to malloc cb_arg\n"); 5271 ocs_free(hw->os, mbxdata, SLI4_BMBX_SIZE); 5272 return OCS_HW_RTN_NO_MEMORY; 5273 } 5274 5275 cb_arg->cb = cb; 5276 cb_arg->arg = arg; 5277 5278 /* payload holds the non-embedded portion */ 5279 if (ocs_dma_alloc(hw->os, &cb_arg->payload, sizeof(sli4_res_common_read_transceiver_data_t), 5280 OCS_MIN_DMA_ALIGNMENT)) { 5281 ocs_log_err(hw->os, "Failed to allocate DMA buffer\n"); 5282 ocs_free(hw->os, cb_arg, sizeof(ocs_hw_sfp_cb_arg_t)); 5283 ocs_free(hw->os, mbxdata, SLI4_BMBX_SIZE); 5284 return OCS_HW_RTN_NO_MEMORY; 5285 } 5286 5287 /* Send the HW command */ 5288 if (sli_cmd_common_read_transceiver_data(&hw->sli, mbxdata, SLI4_BMBX_SIZE, page, 5289 &cb_arg->payload)) { 5290 rc = ocs_hw_command(hw, mbxdata, OCS_CMD_NOWAIT, ocs_hw_cb_sfp, cb_arg); 5291 } 5292 5293 if (rc != OCS_HW_RTN_SUCCESS) { 5294 ocs_log_test(hw->os, "READ_TRANSCEIVER_DATA failed with status %d\n", 5295 rc); 5296 ocs_dma_free(hw->os, &cb_arg->payload); 5297 ocs_free(hw->os, cb_arg, sizeof(ocs_hw_sfp_cb_arg_t)); 5298 ocs_free(hw->os, mbxdata, SLI4_BMBX_SIZE); 5299 } 5300 5301 return rc; 5302 } 5303 5304 /** 5305 * @brief Function to retrieve the temperature information. 5306 * 5307 * @param hw Hardware context. 5308 * @param cb Function call upon completion of sending the data (may be NULL). 5309 * @param arg Argument to pass to IO completion function. 5310 * 5311 * @return Returns OCS_HW_RTN_SUCCESS, OCS_HW_RTN_ERROR, or OCS_HW_RTN_NO_MEMORY. 5312 */ 5313 ocs_hw_rtn_e 5314 ocs_hw_get_temperature(ocs_hw_t *hw, ocs_hw_temp_cb_t cb, void *arg) 5315 { 5316 ocs_hw_rtn_e rc = OCS_HW_RTN_ERROR; 5317 ocs_hw_temp_cb_arg_t *cb_arg; 5318 uint8_t *mbxdata; 5319 5320 mbxdata = ocs_malloc(hw->os, SLI4_BMBX_SIZE, OCS_M_ZERO | OCS_M_NOWAIT); 5321 if (mbxdata == NULL) { 5322 ocs_log_err(hw->os, "failed to malloc mbox"); 5323 return OCS_HW_RTN_NO_MEMORY; 5324 } 5325 5326 cb_arg = ocs_malloc(hw->os, sizeof(ocs_hw_temp_cb_arg_t), OCS_M_NOWAIT); 5327 if (cb_arg == NULL) { 5328 ocs_log_err(hw->os, "failed to malloc cb_arg"); 5329 ocs_free(hw->os, mbxdata, SLI4_BMBX_SIZE); 5330 return OCS_HW_RTN_NO_MEMORY; 5331 } 5332 5333 cb_arg->cb = cb; 5334 cb_arg->arg = arg; 5335 5336 if (sli_cmd_dump_type4(&hw->sli, mbxdata, SLI4_BMBX_SIZE, 5337 SLI4_WKI_TAG_SAT_TEM)) { 5338 rc = ocs_hw_command(hw, mbxdata, OCS_CMD_NOWAIT, ocs_hw_cb_temp, cb_arg); 5339 } 5340 5341 if (rc != OCS_HW_RTN_SUCCESS) { 5342 ocs_log_test(hw->os, "DUMP_TYPE4 failed\n"); 5343 ocs_free(hw->os, mbxdata, SLI4_BMBX_SIZE); 5344 ocs_free(hw->os, cb_arg, sizeof(ocs_hw_temp_cb_arg_t)); 5345 } 5346 5347 return rc; 5348 } 5349 5350 /** 5351 * @brief Called when the DUMP command completes. 5352 * 5353 * @par Description 5354 * Get the temperature data out of the response, free the mailbox that was malloc'd 5355 * by ocs_hw_get_temperature(), then call the callback and pass the status and data. 5356 * 5357 * @param hw Hardware context. 5358 * @param status Status field from the mbox completion. 5359 * @param mqe Mailbox response structure. 5360 * @param arg Pointer to a callback function that signals the caller that the command is done. 5361 * The callback function prototype is defined by ocs_hw_temp_cb_t. 5362 * 5363 * @return Returns 0. 5364 */ 5365 static int32_t 5366 ocs_hw_cb_temp(ocs_hw_t *hw, int32_t status, uint8_t *mqe, void *arg) 5367 { 5368 5369 sli4_cmd_dump4_t* mbox_rsp = (sli4_cmd_dump4_t*) mqe; 5370 ocs_hw_temp_cb_arg_t *cb_arg = arg; 5371 uint32_t curr_temp = mbox_rsp->resp_data[0]; /* word 5 */ 5372 uint32_t crit_temp_thrshld = mbox_rsp->resp_data[1]; /* word 6*/ 5373 uint32_t warn_temp_thrshld = mbox_rsp->resp_data[2]; /* word 7 */ 5374 uint32_t norm_temp_thrshld = mbox_rsp->resp_data[3]; /* word 8 */ 5375 uint32_t fan_off_thrshld = mbox_rsp->resp_data[4]; /* word 9 */ 5376 uint32_t fan_on_thrshld = mbox_rsp->resp_data[5]; /* word 10 */ 5377 5378 if (cb_arg) { 5379 if (cb_arg->cb) { 5380 if ((status == 0) && mbox_rsp->hdr.status) { 5381 status = mbox_rsp->hdr.status; 5382 } 5383 cb_arg->cb(status, 5384 curr_temp, 5385 crit_temp_thrshld, 5386 warn_temp_thrshld, 5387 norm_temp_thrshld, 5388 fan_off_thrshld, 5389 fan_on_thrshld, 5390 cb_arg->arg); 5391 } 5392 5393 ocs_free(hw->os, cb_arg, sizeof(ocs_hw_temp_cb_arg_t)); 5394 } 5395 ocs_free(hw->os, mqe, SLI4_BMBX_SIZE); 5396 5397 return 0; 5398 } 5399 5400 /** 5401 * @brief Function to retrieve the link statistics. 5402 * 5403 * @param hw Hardware context. 5404 * @param req_ext_counters If TRUE, then the extended counters will be requested. 5405 * @param clear_overflow_flags If TRUE, then overflow flags will be cleared. 5406 * @param clear_all_counters If TRUE, the counters will be cleared. 5407 * @param cb Function call upon completion of sending the data (may be NULL). 5408 * @param arg Argument to pass to IO completion function. 5409 * 5410 * @return Returns OCS_HW_RTN_SUCCESS, OCS_HW_RTN_ERROR, or OCS_HW_RTN_NO_MEMORY. 5411 */ 5412 ocs_hw_rtn_e 5413 ocs_hw_get_link_stats(ocs_hw_t *hw, 5414 uint8_t req_ext_counters, 5415 uint8_t clear_overflow_flags, 5416 uint8_t clear_all_counters, 5417 ocs_hw_link_stat_cb_t cb, 5418 void *arg) 5419 { 5420 ocs_hw_rtn_e rc = OCS_HW_RTN_ERROR; 5421 ocs_hw_link_stat_cb_arg_t *cb_arg; 5422 uint8_t *mbxdata; 5423 5424 mbxdata = ocs_malloc(hw->os, SLI4_BMBX_SIZE, OCS_M_ZERO | OCS_M_NOWAIT); 5425 if (mbxdata == NULL) { 5426 ocs_log_err(hw->os, "failed to malloc mbox"); 5427 return OCS_HW_RTN_NO_MEMORY; 5428 } 5429 5430 cb_arg = ocs_malloc(hw->os, sizeof(ocs_hw_link_stat_cb_arg_t), OCS_M_NOWAIT); 5431 if (cb_arg == NULL) { 5432 ocs_log_err(hw->os, "failed to malloc cb_arg"); 5433 ocs_free(hw->os, mbxdata, SLI4_BMBX_SIZE); 5434 return OCS_HW_RTN_NO_MEMORY; 5435 } 5436 5437 cb_arg->cb = cb; 5438 cb_arg->arg = arg; 5439 5440 if (sli_cmd_read_link_stats(&hw->sli, mbxdata, SLI4_BMBX_SIZE, 5441 req_ext_counters, 5442 clear_overflow_flags, 5443 clear_all_counters)) { 5444 rc = ocs_hw_command(hw, mbxdata, OCS_CMD_NOWAIT, ocs_hw_cb_link_stat, cb_arg); 5445 } 5446 5447 if (rc != OCS_HW_RTN_SUCCESS) { 5448 ocs_log_test(hw->os, "READ_LINK_STATS failed\n"); 5449 ocs_free(hw->os, mbxdata, SLI4_BMBX_SIZE); 5450 ocs_free(hw->os, cb_arg, sizeof(ocs_hw_link_stat_cb_arg_t)); 5451 } 5452 5453 return rc; 5454 } 5455 5456 /** 5457 * @brief Called when the READ_LINK_STAT command completes. 5458 * 5459 * @par Description 5460 * Get the counters out of the response, free the mailbox that was malloc'd 5461 * by ocs_hw_get_link_stats(), then call the callback and pass the status and data. 5462 * 5463 * @param hw Hardware context. 5464 * @param status Status field from the mbox completion. 5465 * @param mqe Mailbox response structure. 5466 * @param arg Pointer to a callback function that signals the caller that the command is done. 5467 * The callback function prototype is defined by ocs_hw_link_stat_cb_t. 5468 * 5469 * @return Returns 0. 5470 */ 5471 static int32_t 5472 ocs_hw_cb_link_stat(ocs_hw_t *hw, int32_t status, uint8_t *mqe, void *arg) 5473 { 5474 5475 sli4_cmd_read_link_stats_t* mbox_rsp = (sli4_cmd_read_link_stats_t*) mqe; 5476 ocs_hw_link_stat_cb_arg_t *cb_arg = arg; 5477 ocs_hw_link_stat_counts_t counts[OCS_HW_LINK_STAT_MAX]; 5478 uint32_t num_counters = (mbox_rsp->gec ? 20 : 13); 5479 5480 ocs_memset(counts, 0, sizeof(ocs_hw_link_stat_counts_t) * 5481 OCS_HW_LINK_STAT_MAX); 5482 5483 counts[OCS_HW_LINK_STAT_LINK_FAILURE_COUNT].overflow = mbox_rsp->w02of; 5484 counts[OCS_HW_LINK_STAT_LOSS_OF_SYNC_COUNT].overflow = mbox_rsp->w03of; 5485 counts[OCS_HW_LINK_STAT_LOSS_OF_SIGNAL_COUNT].overflow = mbox_rsp->w04of; 5486 counts[OCS_HW_LINK_STAT_PRIMITIVE_SEQ_COUNT].overflow = mbox_rsp->w05of; 5487 counts[OCS_HW_LINK_STAT_INVALID_XMIT_WORD_COUNT].overflow = mbox_rsp->w06of; 5488 counts[OCS_HW_LINK_STAT_CRC_COUNT].overflow = mbox_rsp->w07of; 5489 counts[OCS_HW_LINK_STAT_PRIMITIVE_SEQ_TIMEOUT_COUNT].overflow = mbox_rsp->w08of; 5490 counts[OCS_HW_LINK_STAT_ELASTIC_BUFFER_OVERRUN_COUNT].overflow = mbox_rsp->w09of; 5491 counts[OCS_HW_LINK_STAT_ARB_TIMEOUT_COUNT].overflow = mbox_rsp->w10of; 5492 counts[OCS_HW_LINK_STAT_ADVERTISED_RCV_B2B_CREDIT].overflow = mbox_rsp->w11of; 5493 counts[OCS_HW_LINK_STAT_CURR_RCV_B2B_CREDIT].overflow = mbox_rsp->w12of; 5494 counts[OCS_HW_LINK_STAT_ADVERTISED_XMIT_B2B_CREDIT].overflow = mbox_rsp->w13of; 5495 counts[OCS_HW_LINK_STAT_CURR_XMIT_B2B_CREDIT].overflow = mbox_rsp->w14of; 5496 counts[OCS_HW_LINK_STAT_RCV_EOFA_COUNT].overflow = mbox_rsp->w15of; 5497 counts[OCS_HW_LINK_STAT_RCV_EOFDTI_COUNT].overflow = mbox_rsp->w16of; 5498 counts[OCS_HW_LINK_STAT_RCV_EOFNI_COUNT].overflow = mbox_rsp->w17of; 5499 counts[OCS_HW_LINK_STAT_RCV_SOFF_COUNT].overflow = mbox_rsp->w18of; 5500 counts[OCS_HW_LINK_STAT_RCV_DROPPED_NO_AER_COUNT].overflow = mbox_rsp->w19of; 5501 counts[OCS_HW_LINK_STAT_RCV_DROPPED_NO_RPI_COUNT].overflow = mbox_rsp->w20of; 5502 counts[OCS_HW_LINK_STAT_RCV_DROPPED_NO_XRI_COUNT].overflow = mbox_rsp->w21of; 5503 5504 counts[OCS_HW_LINK_STAT_LINK_FAILURE_COUNT].counter = mbox_rsp->link_failure_error_count; 5505 counts[OCS_HW_LINK_STAT_LOSS_OF_SYNC_COUNT].counter = mbox_rsp->loss_of_sync_error_count; 5506 counts[OCS_HW_LINK_STAT_LOSS_OF_SIGNAL_COUNT].counter = mbox_rsp->loss_of_signal_error_count; 5507 counts[OCS_HW_LINK_STAT_PRIMITIVE_SEQ_COUNT].counter = mbox_rsp->primitive_sequence_error_count; 5508 counts[OCS_HW_LINK_STAT_INVALID_XMIT_WORD_COUNT].counter = mbox_rsp->invalid_transmission_word_error_count; 5509 counts[OCS_HW_LINK_STAT_CRC_COUNT].counter = mbox_rsp->crc_error_count; 5510 counts[OCS_HW_LINK_STAT_PRIMITIVE_SEQ_TIMEOUT_COUNT].counter = mbox_rsp->primitive_sequence_event_timeout_count; 5511 counts[OCS_HW_LINK_STAT_ELASTIC_BUFFER_OVERRUN_COUNT].counter = mbox_rsp->elastic_buffer_overrun_error_count; 5512 counts[OCS_HW_LINK_STAT_ARB_TIMEOUT_COUNT].counter = mbox_rsp->arbitration_fc_al_timout_count; 5513 counts[OCS_HW_LINK_STAT_ADVERTISED_RCV_B2B_CREDIT].counter = mbox_rsp->advertised_receive_bufftor_to_buffer_credit; 5514 counts[OCS_HW_LINK_STAT_CURR_RCV_B2B_CREDIT].counter = mbox_rsp->current_receive_buffer_to_buffer_credit; 5515 counts[OCS_HW_LINK_STAT_ADVERTISED_XMIT_B2B_CREDIT].counter = mbox_rsp->advertised_transmit_buffer_to_buffer_credit; 5516 counts[OCS_HW_LINK_STAT_CURR_XMIT_B2B_CREDIT].counter = mbox_rsp->current_transmit_buffer_to_buffer_credit; 5517 counts[OCS_HW_LINK_STAT_RCV_EOFA_COUNT].counter = mbox_rsp->received_eofa_count; 5518 counts[OCS_HW_LINK_STAT_RCV_EOFDTI_COUNT].counter = mbox_rsp->received_eofdti_count; 5519 counts[OCS_HW_LINK_STAT_RCV_EOFNI_COUNT].counter = mbox_rsp->received_eofni_count; 5520 counts[OCS_HW_LINK_STAT_RCV_SOFF_COUNT].counter = mbox_rsp->received_soff_count; 5521 counts[OCS_HW_LINK_STAT_RCV_DROPPED_NO_AER_COUNT].counter = mbox_rsp->received_dropped_no_aer_count; 5522 counts[OCS_HW_LINK_STAT_RCV_DROPPED_NO_RPI_COUNT].counter = mbox_rsp->received_dropped_no_available_rpi_resources_count; 5523 counts[OCS_HW_LINK_STAT_RCV_DROPPED_NO_XRI_COUNT].counter = mbox_rsp->received_dropped_no_available_xri_resources_count; 5524 5525 if (cb_arg) { 5526 if (cb_arg->cb) { 5527 if ((status == 0) && mbox_rsp->hdr.status) { 5528 status = mbox_rsp->hdr.status; 5529 } 5530 cb_arg->cb(status, 5531 num_counters, 5532 counts, 5533 cb_arg->arg); 5534 } 5535 5536 ocs_free(hw->os, cb_arg, sizeof(ocs_hw_link_stat_cb_arg_t)); 5537 } 5538 ocs_free(hw->os, mqe, SLI4_BMBX_SIZE); 5539 5540 return 0; 5541 } 5542 5543 /** 5544 * @brief Function to retrieve the link and host statistics. 5545 * 5546 * @param hw Hardware context. 5547 * @param cc clear counters, if TRUE all counters will be cleared. 5548 * @param cb Function call upon completion of receiving the data. 5549 * @param arg Argument to pass to pointer fc hosts statistics structure. 5550 * 5551 * @return Returns OCS_HW_RTN_SUCCESS, OCS_HW_RTN_ERROR, or OCS_HW_RTN_NO_MEMORY. 5552 */ 5553 ocs_hw_rtn_e 5554 ocs_hw_get_host_stats(ocs_hw_t *hw, uint8_t cc, ocs_hw_host_stat_cb_t cb, void *arg) 5555 { 5556 ocs_hw_rtn_e rc = OCS_HW_RTN_ERROR; 5557 ocs_hw_host_stat_cb_arg_t *cb_arg; 5558 uint8_t *mbxdata; 5559 5560 mbxdata = ocs_malloc(hw->os, SLI4_BMBX_SIZE, OCS_M_ZERO); 5561 if (mbxdata == NULL) { 5562 ocs_log_err(hw->os, "failed to malloc mbox"); 5563 return OCS_HW_RTN_NO_MEMORY; 5564 } 5565 5566 cb_arg = ocs_malloc(hw->os, sizeof(ocs_hw_host_stat_cb_arg_t), 0); 5567 if (cb_arg == NULL) { 5568 ocs_log_err(hw->os, "failed to malloc cb_arg"); 5569 ocs_free(hw->os, mbxdata, SLI4_BMBX_SIZE); 5570 return OCS_HW_RTN_NO_MEMORY; 5571 } 5572 5573 cb_arg->cb = cb; 5574 cb_arg->arg = arg; 5575 5576 /* Send the HW command to get the host stats */ 5577 if (sli_cmd_read_status(&hw->sli, mbxdata, SLI4_BMBX_SIZE, cc)) { 5578 rc = ocs_hw_command(hw, mbxdata, OCS_CMD_NOWAIT, ocs_hw_cb_host_stat, cb_arg); 5579 } 5580 5581 if (rc != OCS_HW_RTN_SUCCESS) { 5582 ocs_log_test(hw->os, "READ_HOST_STATS failed\n"); 5583 ocs_free(hw->os, mbxdata, SLI4_BMBX_SIZE); 5584 ocs_free(hw->os, cb_arg, sizeof(ocs_hw_host_stat_cb_arg_t)); 5585 } 5586 5587 return rc; 5588 } 5589 5590 /** 5591 * @brief Called when the READ_STATUS command completes. 5592 * 5593 * @par Description 5594 * Get the counters out of the response, free the mailbox that was malloc'd 5595 * by ocs_hw_get_host_stats(), then call the callback and pass 5596 * the status and data. 5597 * 5598 * @param hw Hardware context. 5599 * @param status Status field from the mbox completion. 5600 * @param mqe Mailbox response structure. 5601 * @param arg Pointer to a callback function that signals the caller that the command is done. 5602 * The callback function prototype is defined by 5603 * ocs_hw_host_stat_cb_t. 5604 * 5605 * @return Returns 0. 5606 */ 5607 static int32_t 5608 ocs_hw_cb_host_stat(ocs_hw_t *hw, int32_t status, uint8_t *mqe, void *arg) 5609 { 5610 5611 sli4_cmd_read_status_t* mbox_rsp = (sli4_cmd_read_status_t*) mqe; 5612 ocs_hw_host_stat_cb_arg_t *cb_arg = arg; 5613 ocs_hw_host_stat_counts_t counts[OCS_HW_HOST_STAT_MAX]; 5614 uint32_t num_counters = OCS_HW_HOST_STAT_MAX; 5615 5616 ocs_memset(counts, 0, sizeof(ocs_hw_host_stat_counts_t) * 5617 OCS_HW_HOST_STAT_MAX); 5618 5619 counts[OCS_HW_HOST_STAT_TX_KBYTE_COUNT].counter = mbox_rsp->transmit_kbyte_count; 5620 counts[OCS_HW_HOST_STAT_RX_KBYTE_COUNT].counter = mbox_rsp->receive_kbyte_count; 5621 counts[OCS_HW_HOST_STAT_TX_FRAME_COUNT].counter = mbox_rsp->transmit_frame_count; 5622 counts[OCS_HW_HOST_STAT_RX_FRAME_COUNT].counter = mbox_rsp->receive_frame_count; 5623 counts[OCS_HW_HOST_STAT_TX_SEQ_COUNT].counter = mbox_rsp->transmit_sequence_count; 5624 counts[OCS_HW_HOST_STAT_RX_SEQ_COUNT].counter = mbox_rsp->receive_sequence_count; 5625 counts[OCS_HW_HOST_STAT_TOTAL_EXCH_ORIG].counter = mbox_rsp->total_exchanges_originator; 5626 counts[OCS_HW_HOST_STAT_TOTAL_EXCH_RESP].counter = mbox_rsp->total_exchanges_responder; 5627 counts[OCS_HW_HOSY_STAT_RX_P_BSY_COUNT].counter = mbox_rsp->receive_p_bsy_count; 5628 counts[OCS_HW_HOST_STAT_RX_F_BSY_COUNT].counter = mbox_rsp->receive_f_bsy_count; 5629 counts[OCS_HW_HOST_STAT_DROP_FRM_DUE_TO_NO_RQ_BUF_COUNT].counter = mbox_rsp->dropped_frames_due_to_no_rq_buffer_count; 5630 counts[OCS_HW_HOST_STAT_EMPTY_RQ_TIMEOUT_COUNT].counter = mbox_rsp->empty_rq_timeout_count; 5631 counts[OCS_HW_HOST_STAT_DROP_FRM_DUE_TO_NO_XRI_COUNT].counter = mbox_rsp->dropped_frames_due_to_no_xri_count; 5632 counts[OCS_HW_HOST_STAT_EMPTY_XRI_POOL_COUNT].counter = mbox_rsp->empty_xri_pool_count; 5633 5634 if (cb_arg) { 5635 if (cb_arg->cb) { 5636 if ((status == 0) && mbox_rsp->hdr.status) { 5637 status = mbox_rsp->hdr.status; 5638 } 5639 cb_arg->cb(status, 5640 num_counters, 5641 counts, 5642 cb_arg->arg); 5643 } 5644 5645 ocs_free(hw->os, cb_arg, sizeof(ocs_hw_host_stat_cb_arg_t)); 5646 } 5647 ocs_free(hw->os, mqe, SLI4_BMBX_SIZE); 5648 5649 return 0; 5650 } 5651 5652 /** 5653 * @brief HW link configuration enum to the CLP string value mapping. 5654 * 5655 * This structure provides a mapping from the ocs_hw_linkcfg_e 5656 * enum (enum exposed for the OCS_HW_PORT_SET_LINK_CONFIG port 5657 * control) to the CLP string that is used 5658 * in the DMTF_CLP_CMD mailbox command. 5659 */ 5660 typedef struct ocs_hw_linkcfg_map_s { 5661 ocs_hw_linkcfg_e linkcfg; 5662 const char *clp_str; 5663 } ocs_hw_linkcfg_map_t; 5664 5665 /** 5666 * @brief Mapping from the HW linkcfg enum to the CLP command value 5667 * string. 5668 */ 5669 static ocs_hw_linkcfg_map_t linkcfg_map[] = { 5670 {OCS_HW_LINKCFG_4X10G, "ELX_4x10G"}, 5671 {OCS_HW_LINKCFG_1X40G, "ELX_1x40G"}, 5672 {OCS_HW_LINKCFG_2X16G, "ELX_2x16G"}, 5673 {OCS_HW_LINKCFG_4X8G, "ELX_4x8G"}, 5674 {OCS_HW_LINKCFG_4X1G, "ELX_4x1G"}, 5675 {OCS_HW_LINKCFG_2X10G, "ELX_2x10G"}, 5676 {OCS_HW_LINKCFG_2X10G_2X8G, "ELX_2x10G_2x8G"}}; 5677 5678 /** 5679 * @brief HW link configuration enum to Skyhawk link config ID mapping. 5680 * 5681 * This structure provides a mapping from the ocs_hw_linkcfg_e 5682 * enum (enum exposed for the OCS_HW_PORT_SET_LINK_CONFIG port 5683 * control) to the link config ID numbers used by Skyhawk 5684 */ 5685 typedef struct ocs_hw_skyhawk_linkcfg_map_s { 5686 ocs_hw_linkcfg_e linkcfg; 5687 uint32_t config_id; 5688 } ocs_hw_skyhawk_linkcfg_map_t; 5689 5690 /** 5691 * @brief Mapping from the HW linkcfg enum to the Skyhawk link config IDs 5692 */ 5693 static ocs_hw_skyhawk_linkcfg_map_t skyhawk_linkcfg_map[] = { 5694 {OCS_HW_LINKCFG_4X10G, 0x0a}, 5695 {OCS_HW_LINKCFG_1X40G, 0x09}, 5696 }; 5697 5698 /** 5699 * @brief Helper function for getting the HW linkcfg enum from the CLP 5700 * string value 5701 * 5702 * @param clp_str CLP string value from OEMELX_LinkConfig. 5703 * 5704 * @return Returns the HW linkcfg enum corresponding to clp_str. 5705 */ 5706 static ocs_hw_linkcfg_e 5707 ocs_hw_linkcfg_from_clp(const char *clp_str) 5708 { 5709 uint32_t i; 5710 for (i = 0; i < ARRAY_SIZE(linkcfg_map); i++) { 5711 if (ocs_strncmp(linkcfg_map[i].clp_str, clp_str, ocs_strlen(clp_str)) == 0) { 5712 return linkcfg_map[i].linkcfg; 5713 } 5714 } 5715 return OCS_HW_LINKCFG_NA; 5716 } 5717 5718 /** 5719 * @brief Helper function for getting the CLP string value from the HW 5720 * linkcfg enum. 5721 * 5722 * @param linkcfg HW linkcfg enum. 5723 * 5724 * @return Returns the OEMELX_LinkConfig CLP string value corresponding to 5725 * given linkcfg. 5726 */ 5727 static const char * 5728 ocs_hw_clp_from_linkcfg(ocs_hw_linkcfg_e linkcfg) 5729 { 5730 uint32_t i; 5731 for (i = 0; i < ARRAY_SIZE(linkcfg_map); i++) { 5732 if (linkcfg_map[i].linkcfg == linkcfg) { 5733 return linkcfg_map[i].clp_str; 5734 } 5735 } 5736 return NULL; 5737 } 5738 5739 /** 5740 * @brief Helper function for getting a Skyhawk link config ID from the HW 5741 * linkcfg enum. 5742 * 5743 * @param linkcfg HW linkcfg enum. 5744 * 5745 * @return Returns the Skyhawk link config ID corresponding to 5746 * given linkcfg. 5747 */ 5748 static uint32_t 5749 ocs_hw_config_id_from_linkcfg(ocs_hw_linkcfg_e linkcfg) 5750 { 5751 uint32_t i; 5752 for (i = 0; i < ARRAY_SIZE(skyhawk_linkcfg_map); i++) { 5753 if (skyhawk_linkcfg_map[i].linkcfg == linkcfg) { 5754 return skyhawk_linkcfg_map[i].config_id; 5755 } 5756 } 5757 return 0; 5758 } 5759 5760 /** 5761 * @brief Helper function for getting the HW linkcfg enum from a 5762 * Skyhawk config ID. 5763 * 5764 * @param config_id Skyhawk link config ID. 5765 * 5766 * @return Returns the HW linkcfg enum corresponding to config_id. 5767 */ 5768 static ocs_hw_linkcfg_e 5769 ocs_hw_linkcfg_from_config_id(const uint32_t config_id) 5770 { 5771 uint32_t i; 5772 for (i = 0; i < ARRAY_SIZE(skyhawk_linkcfg_map); i++) { 5773 if (skyhawk_linkcfg_map[i].config_id == config_id) { 5774 return skyhawk_linkcfg_map[i].linkcfg; 5775 } 5776 } 5777 return OCS_HW_LINKCFG_NA; 5778 } 5779 5780 /** 5781 * @brief Link configuration callback argument. 5782 */ 5783 typedef struct ocs_hw_linkcfg_cb_arg_s { 5784 ocs_hw_port_control_cb_t cb; 5785 void *arg; 5786 uint32_t opts; 5787 int32_t status; 5788 ocs_dma_t dma_cmd; 5789 ocs_dma_t dma_resp; 5790 uint32_t result_len; 5791 } ocs_hw_linkcfg_cb_arg_t; 5792 5793 /** 5794 * @brief Set link configuration. 5795 * 5796 * @param hw Hardware context. 5797 * @param value Link configuration enum to which the link configuration is 5798 * set. 5799 * @param opts Mailbox command options (OCS_CMD_NOWAIT/POLL). 5800 * @param cb Callback function to invoke following mbx command. 5801 * @param arg Callback argument. 5802 * 5803 * @return Returns OCS_HW_RTN_SUCCESS on success. 5804 */ 5805 static ocs_hw_rtn_e 5806 ocs_hw_set_linkcfg(ocs_hw_t *hw, ocs_hw_linkcfg_e value, uint32_t opts, ocs_hw_port_control_cb_t cb, void *arg) 5807 { 5808 if (!sli_link_is_configurable(&hw->sli)) { 5809 ocs_log_debug(hw->os, "Function not supported\n"); 5810 return OCS_HW_RTN_ERROR; 5811 } 5812 5813 if (SLI4_IF_TYPE_LANCER_FC_ETH == sli_get_if_type(&hw->sli)) { 5814 return ocs_hw_set_linkcfg_lancer(hw, value, opts, cb, arg); 5815 } else if ((SLI4_IF_TYPE_BE3_SKH_PF == sli_get_if_type(&hw->sli)) || 5816 (SLI4_IF_TYPE_BE3_SKH_VF == sli_get_if_type(&hw->sli))) { 5817 return ocs_hw_set_linkcfg_skyhawk(hw, value, opts, cb, arg); 5818 } else { 5819 ocs_log_test(hw->os, "Function not supported for this IF_TYPE\n"); 5820 return OCS_HW_RTN_ERROR; 5821 } 5822 } 5823 5824 /** 5825 * @brief Set link configuration for Lancer 5826 * 5827 * @param hw Hardware context. 5828 * @param value Link configuration enum to which the link configuration is 5829 * set. 5830 * @param opts Mailbox command options (OCS_CMD_NOWAIT/POLL). 5831 * @param cb Callback function to invoke following mbx command. 5832 * @param arg Callback argument. 5833 * 5834 * @return Returns OCS_HW_RTN_SUCCESS on success. 5835 */ 5836 static ocs_hw_rtn_e 5837 ocs_hw_set_linkcfg_lancer(ocs_hw_t *hw, ocs_hw_linkcfg_e value, uint32_t opts, ocs_hw_port_control_cb_t cb, void *arg) 5838 { 5839 char cmd[OCS_HW_DMTF_CLP_CMD_MAX]; 5840 ocs_hw_linkcfg_cb_arg_t *cb_arg; 5841 const char *value_str = NULL; 5842 ocs_hw_rtn_e rc = OCS_HW_RTN_SUCCESS; 5843 5844 /* translate ocs_hw_linkcfg_e to CLP string */ 5845 value_str = ocs_hw_clp_from_linkcfg(value); 5846 5847 /* allocate memory for callback argument */ 5848 cb_arg = ocs_malloc(hw->os, sizeof(*cb_arg), OCS_M_NOWAIT); 5849 if (cb_arg == NULL) { 5850 ocs_log_err(hw->os, "failed to malloc cb_arg"); 5851 return OCS_HW_RTN_NO_MEMORY; 5852 } 5853 5854 ocs_snprintf(cmd, OCS_HW_DMTF_CLP_CMD_MAX, "set / OEMELX_LinkConfig=%s", value_str); 5855 /* allocate DMA for command */ 5856 if (ocs_dma_alloc(hw->os, &cb_arg->dma_cmd, ocs_strlen(cmd)+1, 4096)) { 5857 ocs_log_err(hw->os, "malloc failed\n"); 5858 ocs_free(hw->os, cb_arg, sizeof(*cb_arg)); 5859 return OCS_HW_RTN_NO_MEMORY; 5860 } 5861 ocs_memset(cb_arg->dma_cmd.virt, 0, ocs_strlen(cmd)+1); 5862 ocs_memcpy(cb_arg->dma_cmd.virt, cmd, ocs_strlen(cmd)); 5863 5864 /* allocate DMA for response */ 5865 if (ocs_dma_alloc(hw->os, &cb_arg->dma_resp, OCS_HW_DMTF_CLP_RSP_MAX, 4096)) { 5866 ocs_log_err(hw->os, "malloc failed\n"); 5867 ocs_dma_free(hw->os, &cb_arg->dma_cmd); 5868 ocs_free(hw->os, cb_arg, sizeof(*cb_arg)); 5869 return OCS_HW_RTN_NO_MEMORY; 5870 } 5871 cb_arg->cb = cb; 5872 cb_arg->arg = arg; 5873 cb_arg->opts = opts; 5874 5875 rc = ocs_hw_exec_dmtf_clp_cmd(hw, &cb_arg->dma_cmd, &cb_arg->dma_resp, 5876 opts, ocs_hw_linkcfg_dmtf_clp_cb, cb_arg); 5877 5878 if (opts == OCS_CMD_POLL || rc != OCS_HW_RTN_SUCCESS) { 5879 /* if failed, or polling, free memory here; if success and not 5880 * polling, will free in callback function 5881 */ 5882 if (rc) { 5883 ocs_log_test(hw->os, "CLP cmd=\"%s\" failed\n", 5884 (char *)cb_arg->dma_cmd.virt); 5885 } 5886 ocs_dma_free(hw->os, &cb_arg->dma_cmd); 5887 ocs_dma_free(hw->os, &cb_arg->dma_resp); 5888 ocs_free(hw->os, cb_arg, sizeof(*cb_arg)); 5889 } 5890 return rc; 5891 } 5892 5893 /** 5894 * @brief Callback for ocs_hw_set_linkcfg_skyhawk 5895 * 5896 * @param hw Hardware context. 5897 * @param status Status from the RECONFIG_GET_LINK_INFO command. 5898 * @param mqe Mailbox response structure. 5899 * @param arg Pointer to a callback argument. 5900 * 5901 * @return none 5902 */ 5903 static void 5904 ocs_hw_set_active_link_config_cb(ocs_hw_t *hw, int32_t status, uint8_t *mqe, void *arg) 5905 { 5906 ocs_hw_linkcfg_cb_arg_t *cb_arg = (ocs_hw_linkcfg_cb_arg_t *)arg; 5907 5908 if (status) { 5909 ocs_log_test(hw->os, "SET_RECONFIG_LINK_ID failed, status=%d\n", status); 5910 } 5911 5912 /* invoke callback */ 5913 if (cb_arg->cb) { 5914 cb_arg->cb(status, 0, cb_arg->arg); 5915 } 5916 5917 /* if polling, will free memory in calling function */ 5918 if (cb_arg->opts != OCS_CMD_POLL) { 5919 ocs_free(hw->os, cb_arg, sizeof(*cb_arg)); 5920 } 5921 } 5922 5923 /** 5924 * @brief Set link configuration for a Skyhawk 5925 * 5926 * @param hw Hardware context. 5927 * @param value Link configuration enum to which the link configuration is 5928 * set. 5929 * @param opts Mailbox command options (OCS_CMD_NOWAIT/POLL). 5930 * @param cb Callback function to invoke following mbx command. 5931 * @param arg Callback argument. 5932 * 5933 * @return Returns OCS_HW_RTN_SUCCESS on success. 5934 */ 5935 static ocs_hw_rtn_e 5936 ocs_hw_set_linkcfg_skyhawk(ocs_hw_t *hw, ocs_hw_linkcfg_e value, uint32_t opts, ocs_hw_port_control_cb_t cb, void *arg) 5937 { 5938 uint8_t *mbxdata; 5939 ocs_hw_linkcfg_cb_arg_t *cb_arg; 5940 ocs_hw_rtn_e rc = OCS_HW_RTN_SUCCESS; 5941 uint32_t config_id; 5942 5943 config_id = ocs_hw_config_id_from_linkcfg(value); 5944 5945 if (config_id == 0) { 5946 ocs_log_test(hw->os, "Link config %d not supported by Skyhawk\n", value); 5947 return OCS_HW_RTN_ERROR; 5948 } 5949 5950 /* mbxdata holds the header of the command */ 5951 mbxdata = ocs_malloc(hw->os, SLI4_BMBX_SIZE, OCS_M_ZERO | OCS_M_NOWAIT); 5952 if (mbxdata == NULL) { 5953 ocs_log_err(hw->os, "failed to malloc mbox\n"); 5954 return OCS_HW_RTN_NO_MEMORY; 5955 } 5956 5957 /* cb_arg holds the data that will be passed to the callback on completion */ 5958 cb_arg = ocs_malloc(hw->os, sizeof(ocs_hw_linkcfg_cb_arg_t), OCS_M_NOWAIT); 5959 if (cb_arg == NULL) { 5960 ocs_log_err(hw->os, "failed to malloc cb_arg\n"); 5961 ocs_free(hw->os, mbxdata, SLI4_BMBX_SIZE); 5962 return OCS_HW_RTN_NO_MEMORY; 5963 } 5964 5965 cb_arg->cb = cb; 5966 cb_arg->arg = arg; 5967 5968 if (sli_cmd_common_set_reconfig_link_id(&hw->sli, mbxdata, SLI4_BMBX_SIZE, NULL, 0, config_id)) { 5969 rc = ocs_hw_command(hw, mbxdata, opts, ocs_hw_set_active_link_config_cb, cb_arg); 5970 } 5971 5972 if (rc != OCS_HW_RTN_SUCCESS) { 5973 ocs_log_err(hw->os, "SET_RECONFIG_LINK_ID failed\n"); 5974 ocs_free(hw->os, mbxdata, SLI4_BMBX_SIZE); 5975 ocs_free(hw->os, cb_arg, sizeof(ocs_hw_linkcfg_cb_arg_t)); 5976 } else if (opts == OCS_CMD_POLL) { 5977 /* if we're polling we have to call the callback here. */ 5978 ocs_hw_set_active_link_config_cb(hw, 0, mbxdata, cb_arg); 5979 ocs_free(hw->os, mbxdata, SLI4_BMBX_SIZE); 5980 ocs_free(hw->os, cb_arg, sizeof(ocs_hw_linkcfg_cb_arg_t)); 5981 } else { 5982 /* We weren't poling, so the callback got called */ 5983 ocs_free(hw->os, mbxdata, SLI4_BMBX_SIZE); 5984 } 5985 5986 return rc; 5987 } 5988 5989 /** 5990 * @brief Get link configuration. 5991 * 5992 * @param hw Hardware context. 5993 * @param opts Mailbox command options (OCS_CMD_NOWAIT/POLL). 5994 * @param cb Callback function to invoke following mbx command. 5995 * @param arg Callback argument. 5996 * 5997 * @return Returns OCS_HW_RTN_SUCCESS on success. 5998 */ 5999 static ocs_hw_rtn_e 6000 ocs_hw_get_linkcfg(ocs_hw_t *hw, uint32_t opts, ocs_hw_port_control_cb_t cb, void *arg) 6001 { 6002 if (!sli_link_is_configurable(&hw->sli)) { 6003 ocs_log_debug(hw->os, "Function not supported\n"); 6004 return OCS_HW_RTN_ERROR; 6005 } 6006 6007 if ((SLI4_IF_TYPE_LANCER_FC_ETH == sli_get_if_type(&hw->sli)) || 6008 (SLI4_IF_TYPE_LANCER_G7 == sli_get_if_type(&hw->sli))){ 6009 return ocs_hw_get_linkcfg_lancer(hw, opts, cb, arg); 6010 } else if ((SLI4_IF_TYPE_BE3_SKH_PF == sli_get_if_type(&hw->sli)) || 6011 (SLI4_IF_TYPE_BE3_SKH_VF == sli_get_if_type(&hw->sli))) { 6012 return ocs_hw_get_linkcfg_skyhawk(hw, opts, cb, arg); 6013 } else { 6014 ocs_log_test(hw->os, "Function not supported for this IF_TYPE\n"); 6015 return OCS_HW_RTN_ERROR; 6016 } 6017 } 6018 6019 /** 6020 * @brief Get link configuration for a Lancer 6021 * 6022 * @param hw Hardware context. 6023 * @param opts Mailbox command options (OCS_CMD_NOWAIT/POLL). 6024 * @param cb Callback function to invoke following mbx command. 6025 * @param arg Callback argument. 6026 * 6027 * @return Returns OCS_HW_RTN_SUCCESS on success. 6028 */ 6029 static ocs_hw_rtn_e 6030 ocs_hw_get_linkcfg_lancer(ocs_hw_t *hw, uint32_t opts, ocs_hw_port_control_cb_t cb, void *arg) 6031 { 6032 char cmd[OCS_HW_DMTF_CLP_CMD_MAX]; 6033 ocs_hw_linkcfg_cb_arg_t *cb_arg; 6034 ocs_hw_rtn_e rc = OCS_HW_RTN_SUCCESS; 6035 6036 /* allocate memory for callback argument */ 6037 cb_arg = ocs_malloc(hw->os, sizeof(*cb_arg), OCS_M_NOWAIT); 6038 if (cb_arg == NULL) { 6039 ocs_log_err(hw->os, "failed to malloc cb_arg"); 6040 return OCS_HW_RTN_NO_MEMORY; 6041 } 6042 6043 ocs_snprintf(cmd, OCS_HW_DMTF_CLP_CMD_MAX, "show / OEMELX_LinkConfig"); 6044 6045 /* allocate DMA for command */ 6046 if (ocs_dma_alloc(hw->os, &cb_arg->dma_cmd, ocs_strlen(cmd)+1, 4096)) { 6047 ocs_log_err(hw->os, "malloc failed\n"); 6048 ocs_free(hw->os, cb_arg, sizeof(*cb_arg)); 6049 return OCS_HW_RTN_NO_MEMORY; 6050 } 6051 6052 /* copy CLP command to DMA command */ 6053 ocs_memset(cb_arg->dma_cmd.virt, 0, ocs_strlen(cmd)+1); 6054 ocs_memcpy(cb_arg->dma_cmd.virt, cmd, ocs_strlen(cmd)); 6055 6056 /* allocate DMA for response */ 6057 if (ocs_dma_alloc(hw->os, &cb_arg->dma_resp, OCS_HW_DMTF_CLP_RSP_MAX, 4096)) { 6058 ocs_log_err(hw->os, "malloc failed\n"); 6059 ocs_dma_free(hw->os, &cb_arg->dma_cmd); 6060 ocs_free(hw->os, cb_arg, sizeof(*cb_arg)); 6061 return OCS_HW_RTN_NO_MEMORY; 6062 } 6063 cb_arg->cb = cb; 6064 cb_arg->arg = arg; 6065 cb_arg->opts = opts; 6066 6067 rc = ocs_hw_exec_dmtf_clp_cmd(hw, &cb_arg->dma_cmd, &cb_arg->dma_resp, 6068 opts, ocs_hw_linkcfg_dmtf_clp_cb, cb_arg); 6069 6070 if (opts == OCS_CMD_POLL || rc != OCS_HW_RTN_SUCCESS) { 6071 /* if failed or polling, free memory here; if not polling and success, 6072 * will free in callback function 6073 */ 6074 if (rc) { 6075 ocs_log_test(hw->os, "CLP cmd=\"%s\" failed\n", 6076 (char *)cb_arg->dma_cmd.virt); 6077 } 6078 ocs_dma_free(hw->os, &cb_arg->dma_cmd); 6079 ocs_dma_free(hw->os, &cb_arg->dma_resp); 6080 ocs_free(hw->os, cb_arg, sizeof(*cb_arg)); 6081 } 6082 return rc; 6083 } 6084 6085 /** 6086 * @brief Get the link configuration callback. 6087 * 6088 * @param hw Hardware context. 6089 * @param status Status from the RECONFIG_GET_LINK_INFO command. 6090 * @param mqe Mailbox response structure. 6091 * @param arg Pointer to a callback argument. 6092 * 6093 * @return none 6094 */ 6095 static void 6096 ocs_hw_get_active_link_config_cb(ocs_hw_t *hw, int32_t status, uint8_t *mqe, void *arg) 6097 { 6098 ocs_hw_linkcfg_cb_arg_t *cb_arg = (ocs_hw_linkcfg_cb_arg_t *)arg; 6099 sli4_res_common_get_reconfig_link_info_t *rsp = cb_arg->dma_cmd.virt; 6100 ocs_hw_linkcfg_e value = OCS_HW_LINKCFG_NA; 6101 6102 if (status) { 6103 ocs_log_test(hw->os, "GET_RECONFIG_LINK_INFO failed, status=%d\n", status); 6104 } else { 6105 /* Call was successful */ 6106 value = ocs_hw_linkcfg_from_config_id(rsp->active_link_config_id); 6107 } 6108 6109 /* invoke callback */ 6110 if (cb_arg->cb) { 6111 cb_arg->cb(status, value, cb_arg->arg); 6112 } 6113 6114 /* if polling, will free memory in calling function */ 6115 if (cb_arg->opts != OCS_CMD_POLL) { 6116 ocs_dma_free(hw->os, &cb_arg->dma_cmd); 6117 ocs_free(hw->os, cb_arg, sizeof(*cb_arg)); 6118 } 6119 } 6120 6121 /** 6122 * @brief Get link configuration for a Skyhawk. 6123 * 6124 * @param hw Hardware context. 6125 * @param opts Mailbox command options (OCS_CMD_NOWAIT/POLL). 6126 * @param cb Callback function to invoke following mbx command. 6127 * @param arg Callback argument. 6128 * 6129 * @return Returns OCS_HW_RTN_SUCCESS on success. 6130 */ 6131 static ocs_hw_rtn_e 6132 ocs_hw_get_linkcfg_skyhawk(ocs_hw_t *hw, uint32_t opts, ocs_hw_port_control_cb_t cb, void *arg) 6133 { 6134 uint8_t *mbxdata; 6135 ocs_hw_linkcfg_cb_arg_t *cb_arg; 6136 ocs_hw_rtn_e rc = OCS_HW_RTN_SUCCESS; 6137 6138 /* mbxdata holds the header of the command */ 6139 mbxdata = ocs_malloc(hw->os, SLI4_BMBX_SIZE, OCS_M_ZERO | OCS_M_NOWAIT); 6140 if (mbxdata == NULL) { 6141 ocs_log_err(hw->os, "failed to malloc mbox\n"); 6142 return OCS_HW_RTN_NO_MEMORY; 6143 } 6144 6145 /* cb_arg holds the data that will be passed to the callback on completion */ 6146 cb_arg = ocs_malloc(hw->os, sizeof(ocs_hw_linkcfg_cb_arg_t), OCS_M_NOWAIT); 6147 if (cb_arg == NULL) { 6148 ocs_log_err(hw->os, "failed to malloc cb_arg\n"); 6149 ocs_free(hw->os, mbxdata, SLI4_BMBX_SIZE); 6150 return OCS_HW_RTN_NO_MEMORY; 6151 } 6152 6153 cb_arg->cb = cb; 6154 cb_arg->arg = arg; 6155 cb_arg->opts = opts; 6156 6157 /* dma_mem holds the non-embedded portion */ 6158 if (ocs_dma_alloc(hw->os, &cb_arg->dma_cmd, sizeof(sli4_res_common_get_reconfig_link_info_t), 4)) { 6159 ocs_log_err(hw->os, "Failed to allocate DMA buffer\n"); 6160 ocs_free(hw->os, mbxdata, SLI4_BMBX_SIZE); 6161 ocs_free(hw->os, cb_arg, sizeof(ocs_hw_linkcfg_cb_arg_t)); 6162 return OCS_HW_RTN_NO_MEMORY; 6163 } 6164 6165 if (sli_cmd_common_get_reconfig_link_info(&hw->sli, mbxdata, SLI4_BMBX_SIZE, &cb_arg->dma_cmd)) { 6166 rc = ocs_hw_command(hw, mbxdata, opts, ocs_hw_get_active_link_config_cb, cb_arg); 6167 } 6168 6169 if (rc != OCS_HW_RTN_SUCCESS) { 6170 ocs_log_err(hw->os, "GET_RECONFIG_LINK_INFO failed\n"); 6171 ocs_free(hw->os, mbxdata, SLI4_BMBX_SIZE); 6172 ocs_dma_free(hw->os, &cb_arg->dma_cmd); 6173 ocs_free(hw->os, cb_arg, sizeof(ocs_hw_linkcfg_cb_arg_t)); 6174 } else if (opts == OCS_CMD_POLL) { 6175 /* if we're polling we have to call the callback here. */ 6176 ocs_hw_get_active_link_config_cb(hw, 0, mbxdata, cb_arg); 6177 ocs_free(hw->os, mbxdata, SLI4_BMBX_SIZE); 6178 ocs_dma_free(hw->os, &cb_arg->dma_cmd); 6179 ocs_free(hw->os, cb_arg, sizeof(ocs_hw_linkcfg_cb_arg_t)); 6180 } else { 6181 /* We weren't poling, so the callback got called */ 6182 ocs_free(hw->os, mbxdata, SLI4_BMBX_SIZE); 6183 } 6184 6185 return rc; 6186 } 6187 6188 /** 6189 * @brief Sets the DIF seed value. 6190 * 6191 * @param hw Hardware context. 6192 * 6193 * @return Returns OCS_HW_RTN_SUCCESS on success. 6194 */ 6195 static ocs_hw_rtn_e 6196 ocs_hw_set_dif_seed(ocs_hw_t *hw) 6197 { 6198 ocs_hw_rtn_e rc = OCS_HW_RTN_SUCCESS; 6199 uint8_t buf[SLI4_BMBX_SIZE]; 6200 sli4_req_common_set_features_dif_seed_t seed_param; 6201 6202 ocs_memset(&seed_param, 0, sizeof(seed_param)); 6203 seed_param.seed = hw->config.dif_seed; 6204 6205 /* send set_features command */ 6206 if (sli_cmd_common_set_features(&hw->sli, buf, SLI4_BMBX_SIZE, 6207 SLI4_SET_FEATURES_DIF_SEED, 6208 4, 6209 (uint32_t*)&seed_param)) { 6210 rc = ocs_hw_command(hw, buf, OCS_CMD_POLL, NULL, NULL); 6211 if (rc) { 6212 ocs_log_err(hw->os, "ocs_hw_command returns %d\n", rc); 6213 } else { 6214 ocs_log_debug(hw->os, "DIF seed set to 0x%x\n", 6215 hw->config.dif_seed); 6216 } 6217 } else { 6218 ocs_log_err(hw->os, "sli_cmd_common_set_features failed\n"); 6219 rc = OCS_HW_RTN_ERROR; 6220 } 6221 return rc; 6222 } 6223 6224 /** 6225 * @brief Sets the DIF mode value. 6226 * 6227 * @param hw Hardware context. 6228 * 6229 * @return Returns OCS_HW_RTN_SUCCESS on success. 6230 */ 6231 static ocs_hw_rtn_e 6232 ocs_hw_set_dif_mode(ocs_hw_t *hw) 6233 { 6234 ocs_hw_rtn_e rc = OCS_HW_RTN_SUCCESS; 6235 uint8_t buf[SLI4_BMBX_SIZE]; 6236 sli4_req_common_set_features_t10_pi_mem_model_t mode_param; 6237 6238 ocs_memset(&mode_param, 0, sizeof(mode_param)); 6239 mode_param.tmm = (hw->config.dif_mode == OCS_HW_DIF_MODE_INLINE ? 0 : 1); 6240 6241 /* send set_features command */ 6242 if (sli_cmd_common_set_features(&hw->sli, buf, SLI4_BMBX_SIZE, 6243 SLI4_SET_FEATURES_DIF_MEMORY_MODE, 6244 sizeof(mode_param), 6245 (uint32_t*)&mode_param)) { 6246 rc = ocs_hw_command(hw, buf, OCS_CMD_POLL, NULL, NULL); 6247 if (rc) { 6248 ocs_log_err(hw->os, "ocs_hw_command returns %d\n", rc); 6249 } else { 6250 ocs_log_test(hw->os, "DIF mode set to %s\n", 6251 (hw->config.dif_mode == OCS_HW_DIF_MODE_INLINE ? "inline" : "separate")); 6252 } 6253 } else { 6254 ocs_log_err(hw->os, "sli_cmd_common_set_features failed\n"); 6255 rc = OCS_HW_RTN_ERROR; 6256 } 6257 return rc; 6258 } 6259 6260 static void 6261 ocs_hw_watchdog_timer_cb(void *arg) 6262 { 6263 ocs_hw_t *hw = (ocs_hw_t *)arg; 6264 6265 ocs_hw_config_watchdog_timer(hw); 6266 return; 6267 } 6268 6269 static void 6270 ocs_hw_cb_cfg_watchdog(ocs_hw_t *hw, int32_t status, uint8_t *mqe, void *arg) 6271 { 6272 uint16_t timeout = hw->watchdog_timeout; 6273 6274 if (status != 0) { 6275 ocs_log_err(hw->os, "config watchdog timer failed, rc = %d\n", status); 6276 } else { 6277 if(timeout != 0) { 6278 /* keeping callback 500ms before timeout to keep heartbeat alive */ 6279 ocs_setup_timer(hw->os, &hw->watchdog_timer, ocs_hw_watchdog_timer_cb, hw, (timeout*1000 - 500) ); 6280 }else { 6281 ocs_del_timer(&hw->watchdog_timer); 6282 } 6283 } 6284 6285 ocs_free(hw->os, mqe, SLI4_BMBX_SIZE); 6286 return; 6287 } 6288 6289 /** 6290 * @brief Set configuration parameters for watchdog timer feature. 6291 * 6292 * @param hw Hardware context. 6293 * @param timeout Timeout for watchdog timer in seconds 6294 * 6295 * @return Returns OCS_HW_RTN_SUCCESS on success. 6296 */ 6297 static ocs_hw_rtn_e 6298 ocs_hw_config_watchdog_timer(ocs_hw_t *hw) 6299 { 6300 ocs_hw_rtn_e rc = OCS_HW_RTN_SUCCESS; 6301 uint8_t *buf = ocs_malloc(hw->os, SLI4_BMBX_SIZE, OCS_M_NOWAIT); 6302 6303 if (!buf) { 6304 ocs_log_err(hw->os, "no buffer for command\n"); 6305 return OCS_HW_RTN_NO_MEMORY; 6306 } 6307 6308 sli4_cmd_lowlevel_set_watchdog(&hw->sli, buf, SLI4_BMBX_SIZE, hw->watchdog_timeout); 6309 rc = ocs_hw_command(hw, buf, OCS_CMD_NOWAIT, ocs_hw_cb_cfg_watchdog, NULL); 6310 if (rc) { 6311 ocs_free(hw->os, buf, SLI4_BMBX_SIZE); 6312 ocs_log_err(hw->os, "config watchdog timer failed, rc = %d\n", rc); 6313 } 6314 return rc; 6315 } 6316 6317 /** 6318 * @brief Set configuration parameters for auto-generate xfer_rdy T10 PI feature. 6319 * 6320 * @param hw Hardware context. 6321 * @param buf Pointer to a mailbox buffer area. 6322 * 6323 * @return Returns OCS_HW_RTN_SUCCESS on success. 6324 */ 6325 static ocs_hw_rtn_e 6326 ocs_hw_config_auto_xfer_rdy_t10pi(ocs_hw_t *hw, uint8_t *buf) 6327 { 6328 ocs_hw_rtn_e rc = OCS_HW_RTN_SUCCESS; 6329 sli4_req_common_set_features_xfer_rdy_t10pi_t param; 6330 6331 ocs_memset(¶m, 0, sizeof(param)); 6332 param.rtc = (hw->config.auto_xfer_rdy_ref_tag_is_lba ? 0 : 1); 6333 param.atv = (hw->config.auto_xfer_rdy_app_tag_valid ? 1 : 0); 6334 param.tmm = ((hw->config.dif_mode == OCS_HW_DIF_MODE_INLINE) ? 0 : 1); 6335 param.app_tag = hw->config.auto_xfer_rdy_app_tag_value; 6336 param.blk_size = hw->config.auto_xfer_rdy_blk_size_chip; 6337 6338 switch (hw->config.auto_xfer_rdy_p_type) { 6339 case 1: 6340 param.p_type = 0; 6341 break; 6342 case 3: 6343 param.p_type = 2; 6344 break; 6345 default: 6346 ocs_log_err(hw->os, "unsupported p_type %d\n", 6347 hw->config.auto_xfer_rdy_p_type); 6348 return OCS_HW_RTN_ERROR; 6349 } 6350 6351 /* build the set_features command */ 6352 sli_cmd_common_set_features(&hw->sli, buf, SLI4_BMBX_SIZE, 6353 SLI4_SET_FEATURES_SET_CONFIG_AUTO_XFER_RDY_T10PI, 6354 sizeof(param), 6355 ¶m); 6356 6357 rc = ocs_hw_command(hw, buf, OCS_CMD_POLL, NULL, NULL); 6358 if (rc) { 6359 ocs_log_err(hw->os, "ocs_hw_command returns %d\n", rc); 6360 } else { 6361 ocs_log_test(hw->os, "Auto XFER RDY T10 PI configured rtc:%d atv:%d p_type:%d app_tag:%x blk_size:%d\n", 6362 param.rtc, param.atv, param.p_type, 6363 param.app_tag, param.blk_size); 6364 } 6365 6366 return rc; 6367 } 6368 6369 /** 6370 * @brief enable sli port health check 6371 * 6372 * @param hw Hardware context. 6373 * @param buf Pointer to a mailbox buffer area. 6374 * @param query current status of the health check feature enabled/disabled 6375 * @param enable if 1: enable 0: disable 6376 * @param buf Pointer to a mailbox buffer area. 6377 * 6378 * @return Returns OCS_HW_RTN_SUCCESS on success. 6379 */ 6380 static ocs_hw_rtn_e 6381 ocs_hw_config_sli_port_health_check(ocs_hw_t *hw, uint8_t query, uint8_t enable) 6382 { 6383 ocs_hw_rtn_e rc = OCS_HW_RTN_SUCCESS; 6384 uint8_t buf[SLI4_BMBX_SIZE]; 6385 sli4_req_common_set_features_health_check_t param; 6386 6387 ocs_memset(¶m, 0, sizeof(param)); 6388 param.hck = enable; 6389 param.qry = query; 6390 6391 /* build the set_features command */ 6392 sli_cmd_common_set_features(&hw->sli, buf, SLI4_BMBX_SIZE, 6393 SLI4_SET_FEATURES_SLI_PORT_HEALTH_CHECK, 6394 sizeof(param), 6395 ¶m); 6396 6397 rc = ocs_hw_command(hw, buf, OCS_CMD_POLL, NULL, NULL); 6398 if (rc) { 6399 ocs_log_err(hw->os, "ocs_hw_command returns %d\n", rc); 6400 } else { 6401 ocs_log_test(hw->os, "SLI Port Health Check is enabled \n"); 6402 } 6403 6404 return rc; 6405 } 6406 6407 /** 6408 * @brief Set FTD transfer hint feature 6409 * 6410 * @param hw Hardware context. 6411 * @param fdt_xfer_hint size in bytes where read requests are segmented. 6412 * 6413 * @return Returns OCS_HW_RTN_SUCCESS on success. 6414 */ 6415 static ocs_hw_rtn_e 6416 ocs_hw_config_set_fdt_xfer_hint(ocs_hw_t *hw, uint32_t fdt_xfer_hint) 6417 { 6418 ocs_hw_rtn_e rc = OCS_HW_RTN_SUCCESS; 6419 uint8_t buf[SLI4_BMBX_SIZE]; 6420 sli4_req_common_set_features_set_fdt_xfer_hint_t param; 6421 6422 ocs_memset(¶m, 0, sizeof(param)); 6423 param.fdt_xfer_hint = fdt_xfer_hint; 6424 /* build the set_features command */ 6425 sli_cmd_common_set_features(&hw->sli, buf, SLI4_BMBX_SIZE, 6426 SLI4_SET_FEATURES_SET_FTD_XFER_HINT, 6427 sizeof(param), 6428 ¶m); 6429 6430 rc = ocs_hw_command(hw, buf, OCS_CMD_POLL, NULL, NULL); 6431 if (rc) { 6432 ocs_log_warn(hw->os, "set FDT hint %d failed: %d\n", fdt_xfer_hint, rc); 6433 } else { 6434 ocs_log_debug(hw->os, "Set FTD transfer hint to %d\n", param.fdt_xfer_hint); 6435 } 6436 6437 return rc; 6438 } 6439 6440 /** 6441 * @brief Get the link configuration callback. 6442 * 6443 * @param hw Hardware context. 6444 * @param status Status from the DMTF CLP command. 6445 * @param result_len Length, in bytes, of the DMTF CLP result. 6446 * @param arg Pointer to a callback argument. 6447 * 6448 * @return Returns OCS_HW_RTN_SUCCESS on success. 6449 */ 6450 static void 6451 ocs_hw_linkcfg_dmtf_clp_cb(ocs_hw_t *hw, int32_t status, uint32_t result_len, void *arg) 6452 { 6453 int32_t rval; 6454 char retdata_str[64]; 6455 ocs_hw_linkcfg_cb_arg_t *cb_arg = (ocs_hw_linkcfg_cb_arg_t *)arg; 6456 ocs_hw_linkcfg_e linkcfg = OCS_HW_LINKCFG_NA; 6457 6458 if (status) { 6459 ocs_log_test(hw->os, "CLP cmd failed, status=%d\n", status); 6460 } else { 6461 /* parse CLP response to get return data */ 6462 rval = ocs_hw_clp_resp_get_value(hw, "retdata", retdata_str, 6463 sizeof(retdata_str), 6464 cb_arg->dma_resp.virt, 6465 result_len); 6466 6467 if (rval <= 0) { 6468 ocs_log_err(hw->os, "failed to get retdata %d\n", result_len); 6469 } else { 6470 /* translate string into hw enum */ 6471 linkcfg = ocs_hw_linkcfg_from_clp(retdata_str); 6472 } 6473 } 6474 6475 /* invoke callback */ 6476 if (cb_arg->cb) { 6477 cb_arg->cb(status, linkcfg, cb_arg->arg); 6478 } 6479 6480 /* if polling, will free memory in calling function */ 6481 if (cb_arg->opts != OCS_CMD_POLL) { 6482 ocs_dma_free(hw->os, &cb_arg->dma_cmd); 6483 ocs_dma_free(hw->os, &cb_arg->dma_resp); 6484 ocs_free(hw->os, cb_arg, sizeof(*cb_arg)); 6485 } 6486 } 6487 6488 /** 6489 * @brief Set the Lancer dump location 6490 * @par Description 6491 * This function tells a Lancer chip to use a specific DMA 6492 * buffer as a dump location rather than the internal flash. 6493 * 6494 * @param hw Hardware context. 6495 * @param num_buffers The number of DMA buffers to hold the dump (1..n). 6496 * @param dump_buffers DMA buffers to hold the dump. 6497 * 6498 * @return Returns OCS_HW_RTN_SUCCESS on success. 6499 */ 6500 ocs_hw_rtn_e 6501 ocs_hw_set_dump_location(ocs_hw_t *hw, uint32_t num_buffers, ocs_dma_t *dump_buffers, uint8_t fdb) 6502 { 6503 uint8_t bus, dev, func; 6504 ocs_hw_rtn_e rc = OCS_HW_RTN_SUCCESS; 6505 uint8_t buf[SLI4_BMBX_SIZE]; 6506 6507 /* 6508 * Make sure the FW is new enough to support this command. If the FW 6509 * is too old, the FW will UE. 6510 */ 6511 if (hw->workaround.disable_dump_loc) { 6512 ocs_log_test(hw->os, "FW version is too old for this feature\n"); 6513 return OCS_HW_RTN_ERROR; 6514 } 6515 6516 /* This command is only valid for physical port 0 */ 6517 ocs_get_bus_dev_func(hw->os, &bus, &dev, &func); 6518 if (fdb == 0 && func != 0) { 6519 ocs_log_test(hw->os, "function only valid for pci function 0, %d passed\n", 6520 func); 6521 return OCS_HW_RTN_ERROR; 6522 } 6523 6524 /* 6525 * If a single buffer is used, then it may be passed as is to the chip. For multiple buffers, 6526 * We must allocate a SGL list and then pass the address of the list to the chip. 6527 */ 6528 if (num_buffers > 1) { 6529 uint32_t sge_size = num_buffers * sizeof(sli4_sge_t); 6530 sli4_sge_t *sge; 6531 uint32_t i; 6532 6533 if (hw->dump_sges.size < sge_size) { 6534 ocs_dma_free(hw->os, &hw->dump_sges); 6535 if (ocs_dma_alloc(hw->os, &hw->dump_sges, sge_size, OCS_MIN_DMA_ALIGNMENT)) { 6536 ocs_log_err(hw->os, "SGE DMA allocation failed\n"); 6537 return OCS_HW_RTN_NO_MEMORY; 6538 } 6539 } 6540 /* build the SGE list */ 6541 ocs_memset(hw->dump_sges.virt, 0, hw->dump_sges.size); 6542 hw->dump_sges.len = sge_size; 6543 sge = hw->dump_sges.virt; 6544 for (i = 0; i < num_buffers; i++) { 6545 sge[i].buffer_address_high = ocs_addr32_hi(dump_buffers[i].phys); 6546 sge[i].buffer_address_low = ocs_addr32_lo(dump_buffers[i].phys); 6547 sge[i].last = (i == num_buffers - 1 ? 1 : 0); 6548 sge[i].buffer_length = dump_buffers[i].size; 6549 } 6550 rc = sli_cmd_common_set_dump_location(&hw->sli, (void *)buf, 6551 SLI4_BMBX_SIZE, FALSE, TRUE, 6552 &hw->dump_sges, fdb); 6553 } else { 6554 dump_buffers->len = dump_buffers->size; 6555 rc = sli_cmd_common_set_dump_location(&hw->sli, (void *)buf, 6556 SLI4_BMBX_SIZE, FALSE, FALSE, 6557 dump_buffers, fdb); 6558 } 6559 6560 if (rc) { 6561 rc = ocs_hw_command(hw, buf, OCS_CMD_POLL, 6562 NULL, NULL); 6563 if (rc) { 6564 ocs_log_err(hw->os, "ocs_hw_command returns %d\n", 6565 rc); 6566 } 6567 } else { 6568 ocs_log_err(hw->os, 6569 "sli_cmd_common_set_dump_location failed\n"); 6570 rc = OCS_HW_RTN_ERROR; 6571 } 6572 6573 return rc; 6574 } 6575 6576 /** 6577 * @brief Set the Ethernet license. 6578 * 6579 * @par Description 6580 * This function sends the appropriate mailbox command (DMTF 6581 * CLP) to set the Ethernet license to the given license value. 6582 * Since it is used during the time of ocs_hw_init(), the mailbox 6583 * command is sent via polling (the BMBX route). 6584 * 6585 * @param hw Hardware context. 6586 * @param license 32-bit license value. 6587 * 6588 * @return Returns OCS_HW_RTN_SUCCESS on success. 6589 */ 6590 static ocs_hw_rtn_e 6591 ocs_hw_set_eth_license(ocs_hw_t *hw, uint32_t license) 6592 { 6593 ocs_hw_rtn_e rc = OCS_HW_RTN_SUCCESS; 6594 char cmd[OCS_HW_DMTF_CLP_CMD_MAX]; 6595 ocs_dma_t dma_cmd; 6596 ocs_dma_t dma_resp; 6597 6598 /* only for lancer right now */ 6599 if (SLI4_IF_TYPE_LANCER_FC_ETH != sli_get_if_type(&hw->sli)) { 6600 ocs_log_test(hw->os, "Function only supported for I/F type 2\n"); 6601 return OCS_HW_RTN_ERROR; 6602 } 6603 6604 ocs_snprintf(cmd, OCS_HW_DMTF_CLP_CMD_MAX, "set / OEMELX_Ethernet_License=%X", license); 6605 /* allocate DMA for command */ 6606 if (ocs_dma_alloc(hw->os, &dma_cmd, ocs_strlen(cmd)+1, 4096)) { 6607 ocs_log_err(hw->os, "malloc failed\n"); 6608 return OCS_HW_RTN_NO_MEMORY; 6609 } 6610 ocs_memset(dma_cmd.virt, 0, ocs_strlen(cmd)+1); 6611 ocs_memcpy(dma_cmd.virt, cmd, ocs_strlen(cmd)); 6612 6613 /* allocate DMA for response */ 6614 if (ocs_dma_alloc(hw->os, &dma_resp, OCS_HW_DMTF_CLP_RSP_MAX, 4096)) { 6615 ocs_log_err(hw->os, "malloc failed\n"); 6616 ocs_dma_free(hw->os, &dma_cmd); 6617 return OCS_HW_RTN_NO_MEMORY; 6618 } 6619 6620 /* send DMTF CLP command mbx and poll */ 6621 if (ocs_hw_exec_dmtf_clp_cmd(hw, &dma_cmd, &dma_resp, OCS_CMD_POLL, NULL, NULL)) { 6622 ocs_log_err(hw->os, "CLP cmd=\"%s\" failed\n", (char *)dma_cmd.virt); 6623 rc = OCS_HW_RTN_ERROR; 6624 } 6625 6626 ocs_dma_free(hw->os, &dma_cmd); 6627 ocs_dma_free(hw->os, &dma_resp); 6628 return rc; 6629 } 6630 6631 /** 6632 * @brief Callback argument structure for the DMTF CLP commands. 6633 */ 6634 typedef struct ocs_hw_clp_cb_arg_s { 6635 ocs_hw_dmtf_clp_cb_t cb; 6636 ocs_dma_t *dma_resp; 6637 int32_t status; 6638 uint32_t opts; 6639 void *arg; 6640 } ocs_hw_clp_cb_arg_t; 6641 6642 /** 6643 * @brief Execute the DMTF CLP command. 6644 * 6645 * @param hw Hardware context. 6646 * @param dma_cmd DMA buffer containing the CLP command. 6647 * @param dma_resp DMA buffer that will contain the response (if successful). 6648 * @param opts Mailbox command options (such as OCS_CMD_NOWAIT and POLL). 6649 * @param cb Callback function. 6650 * @param arg Callback argument. 6651 * 6652 * @return Returns the number of bytes written to the response 6653 * buffer on success, or a negative value if failed. 6654 */ 6655 static ocs_hw_rtn_e 6656 ocs_hw_exec_dmtf_clp_cmd(ocs_hw_t *hw, ocs_dma_t *dma_cmd, ocs_dma_t *dma_resp, uint32_t opts, ocs_hw_dmtf_clp_cb_t cb, void *arg) 6657 { 6658 ocs_hw_rtn_e rc = OCS_HW_RTN_ERROR; 6659 ocs_hw_clp_cb_arg_t *cb_arg; 6660 uint8_t *mbxdata; 6661 6662 /* allocate DMA for mailbox */ 6663 mbxdata = ocs_malloc(hw->os, SLI4_BMBX_SIZE, OCS_M_ZERO | OCS_M_NOWAIT); 6664 if (mbxdata == NULL) { 6665 ocs_log_err(hw->os, "failed to malloc mbox\n"); 6666 return OCS_HW_RTN_NO_MEMORY; 6667 } 6668 6669 /* allocate memory for callback argument */ 6670 cb_arg = ocs_malloc(hw->os, sizeof(*cb_arg), OCS_M_NOWAIT); 6671 if (cb_arg == NULL) { 6672 ocs_log_err(hw->os, "failed to malloc cb_arg"); 6673 ocs_free(hw->os, mbxdata, SLI4_BMBX_SIZE); 6674 return OCS_HW_RTN_NO_MEMORY; 6675 } 6676 6677 cb_arg->cb = cb; 6678 cb_arg->arg = arg; 6679 cb_arg->dma_resp = dma_resp; 6680 cb_arg->opts = opts; 6681 6682 /* Send the HW command */ 6683 if (sli_cmd_dmtf_exec_clp_cmd(&hw->sli, mbxdata, SLI4_BMBX_SIZE, 6684 dma_cmd, dma_resp)) { 6685 rc = ocs_hw_command(hw, mbxdata, opts, ocs_hw_dmtf_clp_cb, cb_arg); 6686 6687 if (opts == OCS_CMD_POLL && rc == OCS_HW_RTN_SUCCESS) { 6688 /* if we're polling, copy response and invoke callback to 6689 * parse result */ 6690 ocs_memcpy(mbxdata, hw->sli.bmbx.virt, SLI4_BMBX_SIZE); 6691 ocs_hw_dmtf_clp_cb(hw, 0, mbxdata, cb_arg); 6692 6693 /* set rc to resulting or "parsed" status */ 6694 rc = cb_arg->status; 6695 } 6696 6697 /* if failed, or polling, free memory here */ 6698 if (opts == OCS_CMD_POLL || rc != OCS_HW_RTN_SUCCESS) { 6699 if (rc != OCS_HW_RTN_SUCCESS) { 6700 ocs_log_test(hw->os, "ocs_hw_command failed\n"); 6701 } 6702 ocs_free(hw->os, mbxdata, SLI4_BMBX_SIZE); 6703 ocs_free(hw->os, cb_arg, sizeof(*cb_arg)); 6704 } 6705 } else { 6706 ocs_log_test(hw->os, "sli_cmd_dmtf_exec_clp_cmd failed\n"); 6707 rc = OCS_HW_RTN_ERROR; 6708 ocs_free(hw->os, mbxdata, SLI4_BMBX_SIZE); 6709 ocs_free(hw->os, cb_arg, sizeof(*cb_arg)); 6710 } 6711 6712 return rc; 6713 } 6714 6715 /** 6716 * @brief Called when the DMTF CLP command completes. 6717 * 6718 * @param hw Hardware context. 6719 * @param status Status field from the mbox completion. 6720 * @param mqe Mailbox response structure. 6721 * @param arg Pointer to a callback argument. 6722 * 6723 * @return None. 6724 * 6725 */ 6726 static void 6727 ocs_hw_dmtf_clp_cb(ocs_hw_t *hw, int32_t status, uint8_t *mqe, void *arg) 6728 { 6729 int32_t cb_status = 0; 6730 sli4_cmd_sli_config_t* mbox_rsp = (sli4_cmd_sli_config_t*) mqe; 6731 sli4_res_dmtf_exec_clp_cmd_t *clp_rsp = (sli4_res_dmtf_exec_clp_cmd_t *) mbox_rsp->payload.embed; 6732 ocs_hw_clp_cb_arg_t *cb_arg = arg; 6733 uint32_t result_len = 0; 6734 int32_t stat_len; 6735 char stat_str[8]; 6736 6737 /* there are several status codes here, check them all and condense 6738 * into a single callback status 6739 */ 6740 if (status || mbox_rsp->hdr.status || clp_rsp->clp_status) { 6741 ocs_log_debug(hw->os, "status=x%x/x%x/x%x addl=x%x clp=x%x detail=x%x\n", 6742 status, 6743 mbox_rsp->hdr.status, 6744 clp_rsp->hdr.status, 6745 clp_rsp->hdr.additional_status, 6746 clp_rsp->clp_status, 6747 clp_rsp->clp_detailed_status); 6748 if (status) { 6749 cb_status = status; 6750 } else if (mbox_rsp->hdr.status) { 6751 cb_status = mbox_rsp->hdr.status; 6752 } else { 6753 cb_status = clp_rsp->clp_status; 6754 } 6755 } else { 6756 result_len = clp_rsp->resp_length; 6757 } 6758 6759 if (cb_status) { 6760 goto ocs_hw_cb_dmtf_clp_done; 6761 } 6762 6763 if ((result_len == 0) || (cb_arg->dma_resp->size < result_len)) { 6764 ocs_log_test(hw->os, "Invalid response length: resp_len=%zu result len=%d\n", 6765 cb_arg->dma_resp->size, result_len); 6766 cb_status = -1; 6767 goto ocs_hw_cb_dmtf_clp_done; 6768 } 6769 6770 /* parse CLP response to get status */ 6771 stat_len = ocs_hw_clp_resp_get_value(hw, "status", stat_str, 6772 sizeof(stat_str), 6773 cb_arg->dma_resp->virt, 6774 result_len); 6775 6776 if (stat_len <= 0) { 6777 ocs_log_test(hw->os, "failed to get status %d\n", stat_len); 6778 cb_status = -1; 6779 goto ocs_hw_cb_dmtf_clp_done; 6780 } 6781 6782 if (ocs_strcmp(stat_str, "0") != 0) { 6783 ocs_log_test(hw->os, "CLP status indicates failure=%s\n", stat_str); 6784 cb_status = -1; 6785 goto ocs_hw_cb_dmtf_clp_done; 6786 } 6787 6788 ocs_hw_cb_dmtf_clp_done: 6789 6790 /* save status in cb_arg for callers with NULL cb's + polling */ 6791 cb_arg->status = cb_status; 6792 if (cb_arg->cb) { 6793 cb_arg->cb(hw, cb_status, result_len, cb_arg->arg); 6794 } 6795 /* if polling, caller will free memory */ 6796 if (cb_arg->opts != OCS_CMD_POLL) { 6797 ocs_free(hw->os, cb_arg, sizeof(*cb_arg)); 6798 ocs_free(hw->os, mqe, SLI4_BMBX_SIZE); 6799 } 6800 } 6801 6802 /** 6803 * @brief Parse the CLP result and get the value corresponding to the given 6804 * keyword. 6805 * 6806 * @param hw Hardware context. 6807 * @param keyword CLP keyword for which the value is returned. 6808 * @param value Location to which the resulting value is copied. 6809 * @param value_len Length of the value parameter. 6810 * @param resp Pointer to the response buffer that is searched 6811 * for the keyword and value. 6812 * @param resp_len Length of response buffer passed in. 6813 * 6814 * @return Returns the number of bytes written to the value 6815 * buffer on success, or a negative vaue on failure. 6816 */ 6817 static int32_t 6818 ocs_hw_clp_resp_get_value(ocs_hw_t *hw, const char *keyword, char *value, uint32_t value_len, const char *resp, uint32_t resp_len) 6819 { 6820 char *start = NULL; 6821 char *end = NULL; 6822 6823 /* look for specified keyword in string */ 6824 start = ocs_strstr(resp, keyword); 6825 if (start == NULL) { 6826 ocs_log_test(hw->os, "could not find keyword=%s in CLP response\n", 6827 keyword); 6828 return -1; 6829 } 6830 6831 /* now look for '=' and go one past */ 6832 start = ocs_strchr(start, '='); 6833 if (start == NULL) { 6834 ocs_log_test(hw->os, "could not find \'=\' in CLP response for keyword=%s\n", 6835 keyword); 6836 return -1; 6837 } 6838 start++; 6839 6840 /* \r\n terminates value */ 6841 end = ocs_strstr(start, "\r\n"); 6842 if (end == NULL) { 6843 ocs_log_test(hw->os, "could not find \\r\\n for keyword=%s in CLP response\n", 6844 keyword); 6845 return -1; 6846 } 6847 6848 /* make sure given result array is big enough */ 6849 if ((end - start + 1) > value_len) { 6850 ocs_log_test(hw->os, "value len=%d not large enough for actual=%ld\n", 6851 value_len, (end-start)); 6852 return -1; 6853 } 6854 6855 ocs_strncpy(value, start, (end - start)); 6856 value[end-start] = '\0'; 6857 return (end-start+1); 6858 } 6859 6860 /** 6861 * @brief Cause chip to enter an unrecoverable error state. 6862 * 6863 * @par Description 6864 * Cause chip to enter an unrecoverable error state. This is 6865 * used when detecting unexpected FW behavior so that the FW can be 6866 * hwted from the driver as soon as the error is detected. 6867 * 6868 * @param hw Hardware context. 6869 * @param dump Generate dump as part of reset. 6870 * 6871 * @return Returns 0 on success, or a non-zero value on failure. 6872 * 6873 */ 6874 ocs_hw_rtn_e 6875 ocs_hw_raise_ue(ocs_hw_t *hw, uint8_t dump) 6876 { 6877 ocs_hw_rtn_e rc = OCS_HW_RTN_SUCCESS; 6878 6879 if (sli_raise_ue(&hw->sli, dump) != 0) { 6880 rc = OCS_HW_RTN_ERROR; 6881 } else { 6882 if (hw->state != OCS_HW_STATE_UNINITIALIZED) { 6883 hw->state = OCS_HW_STATE_QUEUES_ALLOCATED; 6884 } 6885 } 6886 6887 return rc; 6888 } 6889 6890 /** 6891 * @brief Called when the OBJECT_GET command completes. 6892 * 6893 * @par Description 6894 * Get the number of bytes actually written out of the response, free the mailbox 6895 * that was malloc'd by ocs_hw_dump_get(), then call the callback 6896 * and pass the status and bytes read. 6897 * 6898 * @param hw Hardware context. 6899 * @param status Status field from the mbox completion. 6900 * @param mqe Mailbox response structure. 6901 * @param arg Pointer to a callback function that signals the caller that the command is done. 6902 * The callback function prototype is <tt>void cb(int32_t status, uint32_t bytes_read)</tt>. 6903 * 6904 * @return Returns 0. 6905 */ 6906 static int32_t 6907 ocs_hw_cb_dump_get(ocs_hw_t *hw, int32_t status, uint8_t *mqe, void *arg) 6908 { 6909 sli4_cmd_sli_config_t* mbox_rsp = (sli4_cmd_sli_config_t*) mqe; 6910 sli4_res_common_read_object_t* rd_obj_rsp = (sli4_res_common_read_object_t*) mbox_rsp->payload.embed; 6911 ocs_hw_dump_get_cb_arg_t *cb_arg = arg; 6912 uint32_t bytes_read; 6913 uint8_t eof; 6914 6915 bytes_read = rd_obj_rsp->actual_read_length; 6916 eof = rd_obj_rsp->eof; 6917 6918 if (cb_arg) { 6919 if (cb_arg->cb) { 6920 if ((status == 0) && mbox_rsp->hdr.status) { 6921 status = mbox_rsp->hdr.status; 6922 } 6923 cb_arg->cb(status, bytes_read, eof, cb_arg->arg); 6924 } 6925 6926 ocs_free(hw->os, cb_arg->mbox_cmd, SLI4_BMBX_SIZE); 6927 ocs_free(hw->os, cb_arg, sizeof(ocs_hw_dump_get_cb_arg_t)); 6928 } 6929 6930 return 0; 6931 } 6932 6933 /** 6934 * @brief Read a dump image to the host. 6935 * 6936 * @par Description 6937 * Creates a SLI_CONFIG mailbox command, fills in the correct values to read a 6938 * dump image chunk, then sends the command with the ocs_hw_command(). On completion, 6939 * the callback function ocs_hw_cb_dump_get() gets called to free the mailbox 6940 * and signal the caller that the read has completed. 6941 * 6942 * @param hw Hardware context. 6943 * @param dma DMA structure to transfer the dump chunk into. 6944 * @param size Size of the dump chunk. 6945 * @param offset Offset, in bytes, from the beginning of the dump. 6946 * @param cb Pointer to a callback function that is called when the command completes. 6947 * The callback function prototype is 6948 * <tt>void cb(int32_t status, uint32_t bytes_read, uint8_t eof, void *arg)</tt>. 6949 * @param arg Pointer to be passed to the callback function. 6950 * 6951 * @return Returns 0 on success, or a non-zero value on failure. 6952 */ 6953 ocs_hw_rtn_e 6954 ocs_hw_dump_get(ocs_hw_t *hw, ocs_dma_t *dma, uint32_t size, uint32_t offset, ocs_hw_dump_get_cb_t cb, void *arg) 6955 { 6956 ocs_hw_rtn_e rc = OCS_HW_RTN_ERROR; 6957 uint8_t *mbxdata; 6958 ocs_hw_dump_get_cb_arg_t *cb_arg; 6959 uint32_t opts = (hw->state == OCS_HW_STATE_ACTIVE ? OCS_CMD_NOWAIT : OCS_CMD_POLL); 6960 6961 if (SLI4_IF_TYPE_LANCER_FC_ETH != sli_get_if_type(&hw->sli)) { 6962 ocs_log_test(hw->os, "Function only supported for I/F type 2\n"); 6963 return OCS_HW_RTN_ERROR; 6964 } 6965 6966 if (1 != sli_dump_is_present(&hw->sli)) { 6967 ocs_log_test(hw->os, "No dump is present\n"); 6968 return OCS_HW_RTN_ERROR; 6969 } 6970 6971 if (1 == sli_reset_required(&hw->sli)) { 6972 ocs_log_test(hw->os, "device reset required\n"); 6973 return OCS_HW_RTN_ERROR; 6974 } 6975 6976 mbxdata = ocs_malloc(hw->os, SLI4_BMBX_SIZE, OCS_M_ZERO | OCS_M_NOWAIT); 6977 if (mbxdata == NULL) { 6978 ocs_log_err(hw->os, "failed to malloc mbox\n"); 6979 return OCS_HW_RTN_NO_MEMORY; 6980 } 6981 6982 cb_arg = ocs_malloc(hw->os, sizeof(ocs_hw_dump_get_cb_arg_t), OCS_M_NOWAIT); 6983 if (cb_arg == NULL) { 6984 ocs_log_err(hw->os, "failed to malloc cb_arg\n"); 6985 ocs_free(hw->os, mbxdata, SLI4_BMBX_SIZE); 6986 return OCS_HW_RTN_NO_MEMORY; 6987 } 6988 6989 cb_arg->cb = cb; 6990 cb_arg->arg = arg; 6991 cb_arg->mbox_cmd = mbxdata; 6992 6993 if (sli_cmd_common_read_object(&hw->sli, mbxdata, SLI4_BMBX_SIZE, 6994 size, offset, "/dbg/dump.bin", dma)) { 6995 rc = ocs_hw_command(hw, mbxdata, opts, ocs_hw_cb_dump_get, cb_arg); 6996 if (rc == 0 && opts == OCS_CMD_POLL) { 6997 ocs_memcpy(mbxdata, hw->sli.bmbx.virt, SLI4_BMBX_SIZE); 6998 rc = ocs_hw_cb_dump_get(hw, 0, mbxdata, cb_arg); 6999 } 7000 } 7001 7002 if (rc != OCS_HW_RTN_SUCCESS) { 7003 ocs_log_test(hw->os, "COMMON_READ_OBJECT failed\n"); 7004 ocs_free(hw->os, mbxdata, SLI4_BMBX_SIZE); 7005 ocs_free(hw->os, cb_arg, sizeof(ocs_hw_dump_get_cb_arg_t)); 7006 } 7007 7008 return rc; 7009 } 7010 7011 /** 7012 * @brief Called when the OBJECT_DELETE command completes. 7013 * 7014 * @par Description 7015 * Free the mailbox that was malloc'd 7016 * by ocs_hw_dump_clear(), then call the callback and pass the status. 7017 * 7018 * @param hw Hardware context. 7019 * @param status Status field from the mbox completion. 7020 * @param mqe Mailbox response structure. 7021 * @param arg Pointer to a callback function that signals the caller that the command is done. 7022 * The callback function prototype is <tt>void cb(int32_t status, void *arg)</tt>. 7023 * 7024 * @return Returns 0. 7025 */ 7026 static int32_t 7027 ocs_hw_cb_dump_clear(ocs_hw_t *hw, int32_t status, uint8_t *mqe, void *arg) 7028 { 7029 ocs_hw_dump_clear_cb_arg_t *cb_arg = arg; 7030 sli4_cmd_sli_config_t* mbox_rsp = (sli4_cmd_sli_config_t*) mqe; 7031 7032 if (cb_arg) { 7033 if (cb_arg->cb) { 7034 if ((status == 0) && mbox_rsp->hdr.status) { 7035 status = mbox_rsp->hdr.status; 7036 } 7037 cb_arg->cb(status, cb_arg->arg); 7038 } 7039 7040 ocs_free(hw->os, cb_arg->mbox_cmd, SLI4_BMBX_SIZE); 7041 ocs_free(hw->os, cb_arg, sizeof(ocs_hw_dump_clear_cb_arg_t)); 7042 } 7043 7044 return 0; 7045 } 7046 7047 /** 7048 * @brief Clear a dump image from the device. 7049 * 7050 * @par Description 7051 * Creates a SLI_CONFIG mailbox command, fills it with the correct values to clear 7052 * the dump, then sends the command with ocs_hw_command(). On completion, 7053 * the callback function ocs_hw_cb_dump_clear() gets called to free the mailbox 7054 * and to signal the caller that the write has completed. 7055 * 7056 * @param hw Hardware context. 7057 * @param cb Pointer to a callback function that is called when the command completes. 7058 * The callback function prototype is 7059 * <tt>void cb(int32_t status, uint32_t bytes_written, void *arg)</tt>. 7060 * @param arg Pointer to be passed to the callback function. 7061 * 7062 * @return Returns 0 on success, or a non-zero value on failure. 7063 */ 7064 ocs_hw_rtn_e 7065 ocs_hw_dump_clear(ocs_hw_t *hw, ocs_hw_dump_clear_cb_t cb, void *arg) 7066 { 7067 ocs_hw_rtn_e rc = OCS_HW_RTN_ERROR; 7068 uint8_t *mbxdata; 7069 ocs_hw_dump_clear_cb_arg_t *cb_arg; 7070 uint32_t opts = (hw->state == OCS_HW_STATE_ACTIVE ? OCS_CMD_NOWAIT : OCS_CMD_POLL); 7071 7072 if (SLI4_IF_TYPE_LANCER_FC_ETH != sli_get_if_type(&hw->sli)) { 7073 ocs_log_test(hw->os, "Function only supported for I/F type 2\n"); 7074 return OCS_HW_RTN_ERROR; 7075 } 7076 7077 mbxdata = ocs_malloc(hw->os, SLI4_BMBX_SIZE, OCS_M_ZERO | OCS_M_NOWAIT); 7078 if (mbxdata == NULL) { 7079 ocs_log_err(hw->os, "failed to malloc mbox\n"); 7080 return OCS_HW_RTN_NO_MEMORY; 7081 } 7082 7083 cb_arg = ocs_malloc(hw->os, sizeof(ocs_hw_dump_clear_cb_arg_t), OCS_M_NOWAIT); 7084 if (cb_arg == NULL) { 7085 ocs_log_err(hw->os, "failed to malloc cb_arg\n"); 7086 ocs_free(hw->os, mbxdata, SLI4_BMBX_SIZE); 7087 return OCS_HW_RTN_NO_MEMORY; 7088 } 7089 7090 cb_arg->cb = cb; 7091 cb_arg->arg = arg; 7092 cb_arg->mbox_cmd = mbxdata; 7093 7094 if (sli_cmd_common_delete_object(&hw->sli, mbxdata, SLI4_BMBX_SIZE, 7095 "/dbg/dump.bin")) { 7096 rc = ocs_hw_command(hw, mbxdata, opts, ocs_hw_cb_dump_clear, cb_arg); 7097 if (rc == 0 && opts == OCS_CMD_POLL) { 7098 ocs_memcpy(mbxdata, hw->sli.bmbx.virt, SLI4_BMBX_SIZE); 7099 rc = ocs_hw_cb_dump_clear(hw, 0, mbxdata, cb_arg); 7100 } 7101 } 7102 7103 if (rc != OCS_HW_RTN_SUCCESS) { 7104 ocs_log_test(hw->os, "COMMON_DELETE_OBJECT failed\n"); 7105 ocs_free(hw->os, mbxdata, SLI4_BMBX_SIZE); 7106 ocs_free(hw->os, cb_arg, sizeof(ocs_hw_dump_clear_cb_arg_t)); 7107 } 7108 7109 return rc; 7110 } 7111 7112 typedef struct ocs_hw_get_port_protocol_cb_arg_s { 7113 ocs_get_port_protocol_cb_t cb; 7114 void *arg; 7115 uint32_t pci_func; 7116 ocs_dma_t payload; 7117 } ocs_hw_get_port_protocol_cb_arg_t; 7118 7119 /** 7120 * @brief Called for the completion of get_port_profile for a 7121 * user request. 7122 * 7123 * @param hw Hardware context. 7124 * @param status The status from the MQE. 7125 * @param mqe Pointer to mailbox command buffer. 7126 * @param arg Pointer to a callback argument. 7127 * 7128 * @return Returns 0 on success, or a non-zero value on failure. 7129 */ 7130 static int32_t 7131 ocs_hw_get_port_protocol_cb(ocs_hw_t *hw, int32_t status, 7132 uint8_t *mqe, void *arg) 7133 { 7134 ocs_hw_get_port_protocol_cb_arg_t *cb_arg = arg; 7135 ocs_dma_t *payload = &(cb_arg->payload); 7136 sli4_res_common_get_profile_config_t* response = (sli4_res_common_get_profile_config_t*) payload->virt; 7137 ocs_hw_port_protocol_e port_protocol; 7138 int num_descriptors; 7139 sli4_resource_descriptor_v1_t *desc_p; 7140 sli4_pcie_resource_descriptor_v1_t *pcie_desc_p; 7141 int i; 7142 7143 port_protocol = OCS_HW_PORT_PROTOCOL_OTHER; 7144 7145 num_descriptors = response->desc_count; 7146 desc_p = (sli4_resource_descriptor_v1_t *)response->desc; 7147 for (i=0; i<num_descriptors; i++) { 7148 if (desc_p->descriptor_type == SLI4_RESOURCE_DESCRIPTOR_TYPE_PCIE) { 7149 pcie_desc_p = (sli4_pcie_resource_descriptor_v1_t*) desc_p; 7150 if (pcie_desc_p->pf_number == cb_arg->pci_func) { 7151 switch(pcie_desc_p->pf_type) { 7152 case 0x02: 7153 port_protocol = OCS_HW_PORT_PROTOCOL_ISCSI; 7154 break; 7155 case 0x04: 7156 port_protocol = OCS_HW_PORT_PROTOCOL_FCOE; 7157 break; 7158 case 0x10: 7159 port_protocol = OCS_HW_PORT_PROTOCOL_FC; 7160 break; 7161 default: 7162 port_protocol = OCS_HW_PORT_PROTOCOL_OTHER; 7163 break; 7164 } 7165 } 7166 } 7167 7168 desc_p = (sli4_resource_descriptor_v1_t *) ((uint8_t *)desc_p + desc_p->descriptor_length); 7169 } 7170 7171 if (cb_arg->cb) { 7172 cb_arg->cb(status, port_protocol, cb_arg->arg); 7173 } 7174 7175 ocs_dma_free(hw->os, &cb_arg->payload); 7176 ocs_free(hw->os, cb_arg, sizeof(ocs_hw_get_port_protocol_cb_arg_t)); 7177 ocs_free(hw->os, mqe, SLI4_BMBX_SIZE); 7178 7179 return 0; 7180 } 7181 7182 /** 7183 * @ingroup io 7184 * @brief Get the current port protocol. 7185 * @par Description 7186 * Issues a SLI4 COMMON_GET_PROFILE_CONFIG mailbox. When the 7187 * command completes the provided mgmt callback function is 7188 * called. 7189 * 7190 * @param hw Hardware context. 7191 * @param pci_func PCI function to query for current protocol. 7192 * @param cb Callback function to be called when the command completes. 7193 * @param ul_arg An argument that is passed to the callback function. 7194 * 7195 * @return 7196 * - OCS_HW_RTN_SUCCESS on success. 7197 * - OCS_HW_RTN_NO_MEMORY if a malloc fails. 7198 * - OCS_HW_RTN_NO_RESOURCES if unable to get a command 7199 * context. 7200 * - OCS_HW_RTN_ERROR on any other error. 7201 */ 7202 ocs_hw_rtn_e 7203 ocs_hw_get_port_protocol(ocs_hw_t *hw, uint32_t pci_func, 7204 ocs_get_port_protocol_cb_t cb, void* ul_arg) 7205 { 7206 uint8_t *mbxdata; 7207 ocs_hw_get_port_protocol_cb_arg_t *cb_arg; 7208 ocs_hw_rtn_e rc = OCS_HW_RTN_SUCCESS; 7209 7210 /* Only supported on Skyhawk */ 7211 if (sli_get_if_type(&hw->sli) != SLI4_IF_TYPE_BE3_SKH_PF) { 7212 return OCS_HW_RTN_ERROR; 7213 } 7214 7215 /* mbxdata holds the header of the command */ 7216 mbxdata = ocs_malloc(hw->os, SLI4_BMBX_SIZE, OCS_M_ZERO | OCS_M_NOWAIT); 7217 if (mbxdata == NULL) { 7218 ocs_log_err(hw->os, "failed to malloc mbox\n"); 7219 return OCS_HW_RTN_NO_MEMORY; 7220 } 7221 7222 /* cb_arg holds the data that will be passed to the callback on completion */ 7223 cb_arg = ocs_malloc(hw->os, sizeof(ocs_hw_get_port_protocol_cb_arg_t), OCS_M_NOWAIT); 7224 if (cb_arg == NULL) { 7225 ocs_log_err(hw->os, "failed to malloc cb_arg\n"); 7226 ocs_free(hw->os, mbxdata, SLI4_BMBX_SIZE); 7227 return OCS_HW_RTN_NO_MEMORY; 7228 } 7229 7230 cb_arg->cb = cb; 7231 cb_arg->arg = ul_arg; 7232 cb_arg->pci_func = pci_func; 7233 7234 /* dma_mem holds the non-embedded portion */ 7235 if (ocs_dma_alloc(hw->os, &cb_arg->payload, 4096, 4)) { 7236 ocs_log_err(hw->os, "Failed to allocate DMA buffer\n"); 7237 ocs_free(hw->os, mbxdata, SLI4_BMBX_SIZE); 7238 ocs_free(hw->os, cb_arg, sizeof(ocs_hw_get_port_protocol_cb_arg_t)); 7239 return OCS_HW_RTN_NO_MEMORY; 7240 } 7241 7242 if (sli_cmd_common_get_profile_config(&hw->sli, mbxdata, SLI4_BMBX_SIZE, &cb_arg->payload)) { 7243 rc = ocs_hw_command(hw, mbxdata, OCS_CMD_NOWAIT, ocs_hw_get_port_protocol_cb, cb_arg); 7244 } 7245 7246 if (rc != OCS_HW_RTN_SUCCESS) { 7247 ocs_log_test(hw->os, "GET_PROFILE_CONFIG failed\n"); 7248 ocs_free(hw->os, mbxdata, SLI4_BMBX_SIZE); 7249 ocs_free(hw->os, cb_arg, sizeof(ocs_hw_fw_write_cb_arg_t)); 7250 ocs_dma_free(hw->os, &cb_arg->payload); 7251 } 7252 7253 return rc; 7254 7255 } 7256 7257 typedef struct ocs_hw_set_port_protocol_cb_arg_s { 7258 ocs_set_port_protocol_cb_t cb; 7259 void *arg; 7260 ocs_dma_t payload; 7261 uint32_t new_protocol; 7262 uint32_t pci_func; 7263 } ocs_hw_set_port_protocol_cb_arg_t; 7264 7265 /** 7266 * @brief Called for the completion of set_port_profile for a 7267 * user request. 7268 * 7269 * @par Description 7270 * This is the second of two callbacks for the set_port_protocol 7271 * function. The set operation is a read-modify-write. This 7272 * callback is called when the write (SET_PROFILE_CONFIG) 7273 * completes. 7274 * 7275 * @param hw Hardware context. 7276 * @param status The status from the MQE. 7277 * @param mqe Pointer to mailbox command buffer. 7278 * @param arg Pointer to a callback argument. 7279 * 7280 * @return 0 on success, non-zero otherwise 7281 */ 7282 static int32_t 7283 ocs_hw_set_port_protocol_cb2(ocs_hw_t *hw, int32_t status, uint8_t *mqe, void *arg) 7284 { 7285 ocs_hw_set_port_protocol_cb_arg_t *cb_arg = arg; 7286 7287 if (cb_arg->cb) { 7288 cb_arg->cb( status, cb_arg->arg); 7289 } 7290 7291 ocs_dma_free(hw->os, &(cb_arg->payload)); 7292 ocs_free(hw->os, mqe, SLI4_BMBX_SIZE); 7293 ocs_free(hw->os, arg, sizeof(ocs_hw_set_port_protocol_cb_arg_t)); 7294 7295 return 0; 7296 } 7297 7298 /** 7299 * @brief Called for the completion of set_port_profile for a 7300 * user request. 7301 * 7302 * @par Description 7303 * This is the first of two callbacks for the set_port_protocol 7304 * function. The set operation is a read-modify-write. This 7305 * callback is called when the read completes 7306 * (GET_PROFILE_CONFG). It will updated the resource 7307 * descriptors, then queue the write (SET_PROFILE_CONFIG). 7308 * 7309 * On entry there are three memory areas that were allocated by 7310 * ocs_hw_set_port_protocol. If a failure is detected in this 7311 * function those need to be freed. If this function succeeds 7312 * it allocates three more areas. 7313 * 7314 * @param hw Hardware context. 7315 * @param status The status from the MQE 7316 * @param mqe Pointer to mailbox command buffer. 7317 * @param arg Pointer to a callback argument. 7318 * 7319 * @return Returns 0 on success, or a non-zero value otherwise. 7320 */ 7321 static int32_t 7322 ocs_hw_set_port_protocol_cb1(ocs_hw_t *hw, int32_t status, uint8_t *mqe, void *arg) 7323 { 7324 ocs_hw_set_port_protocol_cb_arg_t *cb_arg = arg; 7325 ocs_dma_t *payload = &(cb_arg->payload); 7326 sli4_res_common_get_profile_config_t* response = (sli4_res_common_get_profile_config_t*) payload->virt; 7327 int num_descriptors; 7328 sli4_resource_descriptor_v1_t *desc_p; 7329 sli4_pcie_resource_descriptor_v1_t *pcie_desc_p; 7330 int i; 7331 ocs_hw_set_port_protocol_cb_arg_t *new_cb_arg; 7332 ocs_hw_port_protocol_e new_protocol; 7333 uint8_t *dst; 7334 sli4_isap_resouce_descriptor_v1_t *isap_desc_p; 7335 uint8_t *mbxdata; 7336 int pci_descriptor_count; 7337 ocs_hw_rtn_e rc = OCS_HW_RTN_SUCCESS; 7338 int num_fcoe_ports = 0; 7339 int num_iscsi_ports = 0; 7340 7341 new_protocol = (ocs_hw_port_protocol_e)cb_arg->new_protocol; 7342 7343 num_descriptors = response->desc_count; 7344 7345 /* Count PCI descriptors */ 7346 pci_descriptor_count = 0; 7347 desc_p = (sli4_resource_descriptor_v1_t *)response->desc; 7348 for (i=0; i<num_descriptors; i++) { 7349 if (desc_p->descriptor_type == SLI4_RESOURCE_DESCRIPTOR_TYPE_PCIE) { 7350 ++pci_descriptor_count; 7351 } 7352 desc_p = (sli4_resource_descriptor_v1_t *) ((uint8_t *)desc_p + desc_p->descriptor_length); 7353 } 7354 7355 /* mbxdata holds the header of the command */ 7356 mbxdata = ocs_malloc(hw->os, SLI4_BMBX_SIZE, OCS_M_ZERO | OCS_M_NOWAIT); 7357 if (mbxdata == NULL) { 7358 ocs_log_err(hw->os, "failed to malloc mbox\n"); 7359 return OCS_HW_RTN_NO_MEMORY; 7360 } 7361 7362 /* cb_arg holds the data that will be passed to the callback on completion */ 7363 new_cb_arg = ocs_malloc(hw->os, sizeof(ocs_hw_set_port_protocol_cb_arg_t), OCS_M_NOWAIT); 7364 if (new_cb_arg == NULL) { 7365 ocs_log_err(hw->os, "failed to malloc cb_arg\n"); 7366 ocs_free(hw->os, mbxdata, SLI4_BMBX_SIZE); 7367 return OCS_HW_RTN_NO_MEMORY; 7368 } 7369 7370 new_cb_arg->cb = cb_arg->cb; 7371 new_cb_arg->arg = cb_arg->arg; 7372 7373 /* Allocate memory for the descriptors we're going to send. This is 7374 * one for each PCI descriptor plus one ISAP descriptor. */ 7375 if (ocs_dma_alloc(hw->os, &new_cb_arg->payload, sizeof(sli4_req_common_set_profile_config_t) + 7376 (pci_descriptor_count * sizeof(sli4_pcie_resource_descriptor_v1_t)) + 7377 sizeof(sli4_isap_resouce_descriptor_v1_t), 4)) { 7378 ocs_log_err(hw->os, "Failed to allocate DMA buffer\n"); 7379 ocs_free(hw->os, mbxdata, SLI4_BMBX_SIZE); 7380 ocs_free(hw->os, new_cb_arg, sizeof(ocs_hw_set_port_protocol_cb_arg_t)); 7381 return OCS_HW_RTN_NO_MEMORY; 7382 } 7383 7384 sli_cmd_common_set_profile_config(&hw->sli, mbxdata, SLI4_BMBX_SIZE, 7385 &new_cb_arg->payload, 7386 0, pci_descriptor_count+1, 1); 7387 7388 /* Point dst to the first descriptor entry in the SET_PROFILE_CONFIG command */ 7389 dst = (uint8_t *)&(((sli4_req_common_set_profile_config_t *) new_cb_arg->payload.virt)->desc); 7390 7391 /* Loop over all descriptors. If the descriptor is a PCIe descriptor, copy it 7392 * to the SET_PROFILE_CONFIG command to be written back. If it's the descriptor 7393 * that we're trying to change also set its pf_type. 7394 */ 7395 desc_p = (sli4_resource_descriptor_v1_t *)response->desc; 7396 for (i=0; i<num_descriptors; i++) { 7397 if (desc_p->descriptor_type == SLI4_RESOURCE_DESCRIPTOR_TYPE_PCIE) { 7398 pcie_desc_p = (sli4_pcie_resource_descriptor_v1_t*) desc_p; 7399 if (pcie_desc_p->pf_number == cb_arg->pci_func) { 7400 /* This is the PCIe descriptor for this OCS instance. 7401 * Update it with the new pf_type */ 7402 switch(new_protocol) { 7403 case OCS_HW_PORT_PROTOCOL_FC: 7404 pcie_desc_p->pf_type = SLI4_PROTOCOL_FC; 7405 break; 7406 case OCS_HW_PORT_PROTOCOL_FCOE: 7407 pcie_desc_p->pf_type = SLI4_PROTOCOL_FCOE; 7408 break; 7409 case OCS_HW_PORT_PROTOCOL_ISCSI: 7410 pcie_desc_p->pf_type = SLI4_PROTOCOL_ISCSI; 7411 break; 7412 default: 7413 pcie_desc_p->pf_type = SLI4_PROTOCOL_DEFAULT; 7414 break; 7415 } 7416 } 7417 7418 if (pcie_desc_p->pf_type == SLI4_PROTOCOL_FCOE) { 7419 ++num_fcoe_ports; 7420 } 7421 if (pcie_desc_p->pf_type == SLI4_PROTOCOL_ISCSI) { 7422 ++num_iscsi_ports; 7423 } 7424 ocs_memcpy(dst, pcie_desc_p, sizeof(sli4_pcie_resource_descriptor_v1_t)); 7425 dst += sizeof(sli4_pcie_resource_descriptor_v1_t); 7426 } 7427 7428 desc_p = (sli4_resource_descriptor_v1_t *) ((uint8_t *)desc_p + desc_p->descriptor_length); 7429 } 7430 7431 /* Create an ISAP resource descriptor */ 7432 isap_desc_p = (sli4_isap_resouce_descriptor_v1_t*)dst; 7433 isap_desc_p->descriptor_type = SLI4_RESOURCE_DESCRIPTOR_TYPE_ISAP; 7434 isap_desc_p->descriptor_length = sizeof(sli4_isap_resouce_descriptor_v1_t); 7435 if (num_iscsi_ports > 0) { 7436 isap_desc_p->iscsi_tgt = 1; 7437 isap_desc_p->iscsi_ini = 1; 7438 isap_desc_p->iscsi_dif = 1; 7439 } 7440 if (num_fcoe_ports > 0) { 7441 isap_desc_p->fcoe_tgt = 1; 7442 isap_desc_p->fcoe_ini = 1; 7443 isap_desc_p->fcoe_dif = 1; 7444 } 7445 7446 /* At this point we're done with the memory allocated by ocs_port_set_protocol */ 7447 ocs_dma_free(hw->os, &cb_arg->payload); 7448 ocs_free(hw->os, mqe, SLI4_BMBX_SIZE); 7449 ocs_free(hw->os, cb_arg, sizeof(ocs_hw_set_port_protocol_cb_arg_t)); 7450 7451 /* Send a SET_PROFILE_CONFIG mailbox command with the new descriptors */ 7452 rc = ocs_hw_command(hw, mbxdata, OCS_CMD_NOWAIT, ocs_hw_set_port_protocol_cb2, new_cb_arg); 7453 if (rc) { 7454 ocs_log_err(hw->os, "Error posting COMMON_SET_PROFILE_CONFIG\n"); 7455 /* Call the upper level callback to report a failure */ 7456 if (new_cb_arg->cb) { 7457 new_cb_arg->cb( rc, new_cb_arg->arg); 7458 } 7459 7460 /* Free the memory allocated by this function */ 7461 ocs_dma_free(hw->os, &new_cb_arg->payload); 7462 ocs_free(hw->os, mbxdata, SLI4_BMBX_SIZE); 7463 ocs_free(hw->os, new_cb_arg, sizeof(ocs_hw_set_port_protocol_cb_arg_t)); 7464 } 7465 7466 return rc; 7467 } 7468 7469 /** 7470 * @ingroup io 7471 * @brief Set the port protocol. 7472 * @par Description 7473 * Setting the port protocol is a read-modify-write operation. 7474 * This function submits a GET_PROFILE_CONFIG command to read 7475 * the current settings. The callback function will modify the 7476 * settings and issue the write. 7477 * 7478 * On successful completion this function will have allocated 7479 * two regular memory areas and one dma area which will need to 7480 * get freed later in the callbacks. 7481 * 7482 * @param hw Hardware context. 7483 * @param new_protocol New protocol to use. 7484 * @param pci_func PCI function to configure. 7485 * @param cb Callback function to be called when the command completes. 7486 * @param ul_arg An argument that is passed to the callback function. 7487 * 7488 * @return 7489 * - OCS_HW_RTN_SUCCESS on success. 7490 * - OCS_HW_RTN_NO_MEMORY if a malloc fails. 7491 * - OCS_HW_RTN_NO_RESOURCES if unable to get a command 7492 * context. 7493 * - OCS_HW_RTN_ERROR on any other error. 7494 */ 7495 ocs_hw_rtn_e 7496 ocs_hw_set_port_protocol(ocs_hw_t *hw, ocs_hw_port_protocol_e new_protocol, 7497 uint32_t pci_func, ocs_set_port_protocol_cb_t cb, void *ul_arg) 7498 { 7499 uint8_t *mbxdata; 7500 ocs_hw_set_port_protocol_cb_arg_t *cb_arg; 7501 ocs_hw_rtn_e rc = OCS_HW_RTN_ERROR; 7502 7503 /* Only supported on Skyhawk */ 7504 if (sli_get_if_type(&hw->sli) != SLI4_IF_TYPE_BE3_SKH_PF) { 7505 return OCS_HW_RTN_ERROR; 7506 } 7507 7508 /* mbxdata holds the header of the command */ 7509 mbxdata = ocs_malloc(hw->os, SLI4_BMBX_SIZE, OCS_M_ZERO | OCS_M_NOWAIT); 7510 if (mbxdata == NULL) { 7511 ocs_log_err(hw->os, "failed to malloc mbox\n"); 7512 return OCS_HW_RTN_NO_MEMORY; 7513 } 7514 7515 /* cb_arg holds the data that will be passed to the callback on completion */ 7516 cb_arg = ocs_malloc(hw->os, sizeof(ocs_hw_set_port_protocol_cb_arg_t), OCS_M_NOWAIT); 7517 if (cb_arg == NULL) { 7518 ocs_log_err(hw->os, "failed to malloc cb_arg\n"); 7519 ocs_free(hw->os, mbxdata, SLI4_BMBX_SIZE); 7520 return OCS_HW_RTN_NO_MEMORY; 7521 } 7522 7523 cb_arg->cb = cb; 7524 cb_arg->arg = ul_arg; 7525 cb_arg->new_protocol = new_protocol; 7526 cb_arg->pci_func = pci_func; 7527 7528 /* dma_mem holds the non-embedded portion */ 7529 if (ocs_dma_alloc(hw->os, &cb_arg->payload, 4096, 4)) { 7530 ocs_log_err(hw->os, "Failed to allocate DMA buffer\n"); 7531 ocs_free(hw->os, mbxdata, SLI4_BMBX_SIZE); 7532 ocs_free(hw->os, cb_arg, sizeof(ocs_hw_get_port_protocol_cb_arg_t)); 7533 return OCS_HW_RTN_NO_MEMORY; 7534 } 7535 7536 if (sli_cmd_common_get_profile_config(&hw->sli, mbxdata, SLI4_BMBX_SIZE, &cb_arg->payload)) { 7537 rc = ocs_hw_command(hw, mbxdata, OCS_CMD_NOWAIT, ocs_hw_set_port_protocol_cb1, cb_arg); 7538 } 7539 7540 if (rc != OCS_HW_RTN_SUCCESS) { 7541 ocs_log_test(hw->os, "GET_PROFILE_CONFIG failed\n"); 7542 ocs_free(hw->os, mbxdata, SLI4_BMBX_SIZE); 7543 ocs_free(hw->os, cb_arg, sizeof(ocs_hw_fw_write_cb_arg_t)); 7544 ocs_dma_free(hw->os, &cb_arg->payload); 7545 } 7546 7547 return rc; 7548 } 7549 7550 typedef struct ocs_hw_get_profile_list_cb_arg_s { 7551 ocs_get_profile_list_cb_t cb; 7552 void *arg; 7553 ocs_dma_t payload; 7554 } ocs_hw_get_profile_list_cb_arg_t; 7555 7556 /** 7557 * @brief Called for the completion of get_profile_list for a 7558 * user request. 7559 * @par Description 7560 * This function is called when the COMMMON_GET_PROFILE_LIST 7561 * mailbox completes. The response will be in 7562 * ctx->non_embedded_mem.virt. This function parses the 7563 * response and creates a ocs_hw_profile_list, then calls the 7564 * mgmt_cb callback function and passes that list to it. 7565 * 7566 * @param hw Hardware context. 7567 * @param status The status from the MQE 7568 * @param mqe Pointer to mailbox command buffer. 7569 * @param arg Pointer to a callback argument. 7570 * 7571 * @return Returns 0 on success, or a non-zero value on failure. 7572 */ 7573 static int32_t 7574 ocs_hw_get_profile_list_cb(ocs_hw_t *hw, int32_t status, uint8_t *mqe, void *arg) 7575 { 7576 ocs_hw_profile_list_t *list; 7577 ocs_hw_get_profile_list_cb_arg_t *cb_arg = arg; 7578 ocs_dma_t *payload = &(cb_arg->payload); 7579 sli4_res_common_get_profile_list_t *response = (sli4_res_common_get_profile_list_t *)payload->virt; 7580 int i; 7581 int num_descriptors; 7582 7583 list = ocs_malloc(hw->os, sizeof(ocs_hw_profile_list_t), OCS_M_ZERO); 7584 if (list == NULL) { 7585 ocs_log_err(hw->os, "failed to malloc list\n"); 7586 return OCS_HW_RTN_NO_MEMORY; 7587 } 7588 7589 list->num_descriptors = response->profile_descriptor_count; 7590 7591 num_descriptors = list->num_descriptors; 7592 if (num_descriptors > OCS_HW_MAX_PROFILES) { 7593 num_descriptors = OCS_HW_MAX_PROFILES; 7594 } 7595 7596 for (i=0; i<num_descriptors; i++) { 7597 list->descriptors[i].profile_id = response->profile_descriptor[i].profile_id; 7598 list->descriptors[i].profile_index = response->profile_descriptor[i].profile_index; 7599 ocs_strcpy(list->descriptors[i].profile_description, (char *)response->profile_descriptor[i].profile_description); 7600 } 7601 7602 if (cb_arg->cb) { 7603 cb_arg->cb(status, list, cb_arg->arg); 7604 } else { 7605 ocs_free(hw->os, list, sizeof(*list)); 7606 } 7607 7608 ocs_free(hw->os, mqe, SLI4_BMBX_SIZE); 7609 ocs_dma_free(hw->os, &cb_arg->payload); 7610 ocs_free(hw->os, cb_arg, sizeof(ocs_hw_get_profile_list_cb_arg_t)); 7611 7612 return 0; 7613 } 7614 7615 /** 7616 * @ingroup io 7617 * @brief Get a list of available profiles. 7618 * @par Description 7619 * Issues a SLI-4 COMMON_GET_PROFILE_LIST mailbox. When the 7620 * command completes the provided mgmt callback function is 7621 * called. 7622 * 7623 * @param hw Hardware context. 7624 * @param cb Callback function to be called when the 7625 * command completes. 7626 * @param ul_arg An argument that is passed to the callback 7627 * function. 7628 * 7629 * @return 7630 * - OCS_HW_RTN_SUCCESS on success. 7631 * - OCS_HW_RTN_NO_MEMORY if a malloc fails. 7632 * - OCS_HW_RTN_NO_RESOURCES if unable to get a command 7633 * context. 7634 * - OCS_HW_RTN_ERROR on any other error. 7635 */ 7636 ocs_hw_rtn_e 7637 ocs_hw_get_profile_list(ocs_hw_t *hw, ocs_get_profile_list_cb_t cb, void* ul_arg) 7638 { 7639 uint8_t *mbxdata; 7640 ocs_hw_get_profile_list_cb_arg_t *cb_arg; 7641 ocs_hw_rtn_e rc = OCS_HW_RTN_SUCCESS; 7642 7643 /* Only supported on Skyhawk */ 7644 if (sli_get_if_type(&hw->sli) != SLI4_IF_TYPE_BE3_SKH_PF) { 7645 return OCS_HW_RTN_ERROR; 7646 } 7647 7648 /* mbxdata holds the header of the command */ 7649 mbxdata = ocs_malloc(hw->os, SLI4_BMBX_SIZE, OCS_M_ZERO | OCS_M_NOWAIT); 7650 if (mbxdata == NULL) { 7651 ocs_log_err(hw->os, "failed to malloc mbox\n"); 7652 return OCS_HW_RTN_NO_MEMORY; 7653 } 7654 7655 /* cb_arg holds the data that will be passed to the callback on completion */ 7656 cb_arg = ocs_malloc(hw->os, sizeof(ocs_hw_get_profile_list_cb_arg_t), OCS_M_NOWAIT); 7657 if (cb_arg == NULL) { 7658 ocs_log_err(hw->os, "failed to malloc cb_arg\n"); 7659 ocs_free(hw->os, mbxdata, SLI4_BMBX_SIZE); 7660 return OCS_HW_RTN_NO_MEMORY; 7661 } 7662 7663 cb_arg->cb = cb; 7664 cb_arg->arg = ul_arg; 7665 7666 /* dma_mem holds the non-embedded portion */ 7667 if (ocs_dma_alloc(hw->os, &cb_arg->payload, sizeof(sli4_res_common_get_profile_list_t), 4)) { 7668 ocs_log_err(hw->os, "Failed to allocate DMA buffer\n"); 7669 ocs_free(hw->os, mbxdata, SLI4_BMBX_SIZE); 7670 ocs_free(hw->os, cb_arg, sizeof(ocs_hw_get_profile_list_cb_arg_t)); 7671 return OCS_HW_RTN_NO_MEMORY; 7672 } 7673 7674 if (sli_cmd_common_get_profile_list(&hw->sli, mbxdata, SLI4_BMBX_SIZE, 0, &cb_arg->payload)) { 7675 rc = ocs_hw_command(hw, mbxdata, OCS_CMD_NOWAIT, ocs_hw_get_profile_list_cb, cb_arg); 7676 } 7677 7678 if (rc != OCS_HW_RTN_SUCCESS) { 7679 ocs_log_test(hw->os, "GET_PROFILE_LIST failed\n"); 7680 ocs_free(hw->os, mbxdata, SLI4_BMBX_SIZE); 7681 ocs_dma_free(hw->os, &cb_arg->payload); 7682 ocs_free(hw->os, cb_arg, sizeof(ocs_hw_get_profile_list_cb_arg_t)); 7683 } 7684 7685 return rc; 7686 } 7687 7688 typedef struct ocs_hw_get_active_profile_cb_arg_s { 7689 ocs_get_active_profile_cb_t cb; 7690 void *arg; 7691 } ocs_hw_get_active_profile_cb_arg_t; 7692 7693 /** 7694 * @brief Called for the completion of get_active_profile for a 7695 * user request. 7696 * 7697 * @param hw Hardware context. 7698 * @param status The status from the MQE 7699 * @param mqe Pointer to mailbox command buffer. 7700 * @param arg Pointer to a callback argument. 7701 * 7702 * @return Returns 0 on success, or a non-zero value on failure. 7703 */ 7704 static int32_t 7705 ocs_hw_get_active_profile_cb(ocs_hw_t *hw, int32_t status, uint8_t *mqe, void *arg) 7706 { 7707 ocs_hw_get_active_profile_cb_arg_t *cb_arg = arg; 7708 sli4_cmd_sli_config_t* mbox_rsp = (sli4_cmd_sli_config_t*) mqe; 7709 sli4_res_common_get_active_profile_t* response = (sli4_res_common_get_active_profile_t*) mbox_rsp->payload.embed; 7710 uint32_t active_profile; 7711 7712 active_profile = response->active_profile_id; 7713 7714 if (cb_arg->cb) { 7715 cb_arg->cb(status, active_profile, cb_arg->arg); 7716 } 7717 7718 ocs_free(hw->os, mqe, SLI4_BMBX_SIZE); 7719 ocs_free(hw->os, cb_arg, sizeof(ocs_hw_get_active_profile_cb_arg_t)); 7720 7721 return 0; 7722 } 7723 7724 /** 7725 * @ingroup io 7726 * @brief Get the currently active profile. 7727 * @par Description 7728 * Issues a SLI-4 COMMON_GET_ACTIVE_PROFILE mailbox. When the 7729 * command completes the provided mgmt callback function is 7730 * called. 7731 * 7732 * @param hw Hardware context. 7733 * @param cb Callback function to be called when the 7734 * command completes. 7735 * @param ul_arg An argument that is passed to the callback 7736 * function. 7737 * 7738 * @return 7739 * - OCS_HW_RTN_SUCCESS on success. 7740 * - OCS_HW_RTN_NO_MEMORY if a malloc fails. 7741 * - OCS_HW_RTN_NO_RESOURCES if unable to get a command 7742 * context. 7743 * - OCS_HW_RTN_ERROR on any other error. 7744 */ 7745 int32_t 7746 ocs_hw_get_active_profile(ocs_hw_t *hw, ocs_get_active_profile_cb_t cb, void* ul_arg) 7747 { 7748 uint8_t *mbxdata; 7749 ocs_hw_get_active_profile_cb_arg_t *cb_arg; 7750 ocs_hw_rtn_e rc = OCS_HW_RTN_SUCCESS; 7751 7752 /* Only supported on Skyhawk */ 7753 if (sli_get_if_type(&hw->sli) != SLI4_IF_TYPE_BE3_SKH_PF) { 7754 return OCS_HW_RTN_ERROR; 7755 } 7756 7757 /* mbxdata holds the header of the command */ 7758 mbxdata = ocs_malloc(hw->os, SLI4_BMBX_SIZE, OCS_M_ZERO | OCS_M_NOWAIT); 7759 if (mbxdata == NULL) { 7760 ocs_log_err(hw->os, "failed to malloc mbox\n"); 7761 return OCS_HW_RTN_NO_MEMORY; 7762 } 7763 7764 /* cb_arg holds the data that will be passed to the callback on completion */ 7765 cb_arg = ocs_malloc(hw->os, sizeof(ocs_hw_get_active_profile_cb_arg_t), OCS_M_NOWAIT); 7766 if (cb_arg == NULL) { 7767 ocs_log_err(hw->os, "failed to malloc cb_arg\n"); 7768 ocs_free(hw->os, mbxdata, SLI4_BMBX_SIZE); 7769 return OCS_HW_RTN_NO_MEMORY; 7770 } 7771 7772 cb_arg->cb = cb; 7773 cb_arg->arg = ul_arg; 7774 7775 if (sli_cmd_common_get_active_profile(&hw->sli, mbxdata, SLI4_BMBX_SIZE)) { 7776 rc = ocs_hw_command(hw, mbxdata, OCS_CMD_NOWAIT, ocs_hw_get_active_profile_cb, cb_arg); 7777 } 7778 7779 if (rc != OCS_HW_RTN_SUCCESS) { 7780 ocs_log_test(hw->os, "GET_ACTIVE_PROFILE failed\n"); 7781 ocs_free(hw->os, mbxdata, SLI4_BMBX_SIZE); 7782 ocs_free(hw->os, cb_arg, sizeof(ocs_hw_get_active_profile_cb_arg_t)); 7783 } 7784 7785 return rc; 7786 } 7787 7788 typedef struct ocs_hw_get_nvparms_cb_arg_s { 7789 ocs_get_nvparms_cb_t cb; 7790 void *arg; 7791 } ocs_hw_get_nvparms_cb_arg_t; 7792 7793 /** 7794 * @brief Called for the completion of get_nvparms for a 7795 * user request. 7796 * 7797 * @param hw Hardware context. 7798 * @param status The status from the MQE. 7799 * @param mqe Pointer to mailbox command buffer. 7800 * @param arg Pointer to a callback argument. 7801 * 7802 * @return 0 on success, non-zero otherwise 7803 */ 7804 static int32_t 7805 ocs_hw_get_nvparms_cb(ocs_hw_t *hw, int32_t status, uint8_t *mqe, void *arg) 7806 { 7807 ocs_hw_get_nvparms_cb_arg_t *cb_arg = arg; 7808 sli4_cmd_read_nvparms_t* mbox_rsp = (sli4_cmd_read_nvparms_t*) mqe; 7809 7810 if (cb_arg->cb) { 7811 cb_arg->cb(status, mbox_rsp->wwpn, mbox_rsp->wwnn, mbox_rsp->hard_alpa, 7812 mbox_rsp->preferred_d_id, cb_arg->arg); 7813 } 7814 7815 ocs_free(hw->os, mqe, SLI4_BMBX_SIZE); 7816 ocs_free(hw->os, cb_arg, sizeof(ocs_hw_get_nvparms_cb_arg_t)); 7817 7818 return 0; 7819 } 7820 7821 /** 7822 * @ingroup io 7823 * @brief Read non-volatile parms. 7824 * @par Description 7825 * Issues a SLI-4 READ_NVPARMS mailbox. When the 7826 * command completes the provided mgmt callback function is 7827 * called. 7828 * 7829 * @param hw Hardware context. 7830 * @param cb Callback function to be called when the 7831 * command completes. 7832 * @param ul_arg An argument that is passed to the callback 7833 * function. 7834 * 7835 * @return 7836 * - OCS_HW_RTN_SUCCESS on success. 7837 * - OCS_HW_RTN_NO_MEMORY if a malloc fails. 7838 * - OCS_HW_RTN_NO_RESOURCES if unable to get a command 7839 * context. 7840 * - OCS_HW_RTN_ERROR on any other error. 7841 */ 7842 int32_t 7843 ocs_hw_get_nvparms(ocs_hw_t *hw, ocs_get_nvparms_cb_t cb, void* ul_arg) 7844 { 7845 uint8_t *mbxdata; 7846 ocs_hw_get_nvparms_cb_arg_t *cb_arg; 7847 ocs_hw_rtn_e rc = OCS_HW_RTN_SUCCESS; 7848 7849 /* mbxdata holds the header of the command */ 7850 mbxdata = ocs_malloc(hw->os, SLI4_BMBX_SIZE, OCS_M_ZERO | OCS_M_NOWAIT); 7851 if (mbxdata == NULL) { 7852 ocs_log_err(hw->os, "failed to malloc mbox\n"); 7853 return OCS_HW_RTN_NO_MEMORY; 7854 } 7855 7856 /* cb_arg holds the data that will be passed to the callback on completion */ 7857 cb_arg = ocs_malloc(hw->os, sizeof(ocs_hw_get_nvparms_cb_arg_t), OCS_M_NOWAIT); 7858 if (cb_arg == NULL) { 7859 ocs_log_err(hw->os, "failed to malloc cb_arg\n"); 7860 ocs_free(hw->os, mbxdata, SLI4_BMBX_SIZE); 7861 return OCS_HW_RTN_NO_MEMORY; 7862 } 7863 7864 cb_arg->cb = cb; 7865 cb_arg->arg = ul_arg; 7866 7867 if (sli_cmd_read_nvparms(&hw->sli, mbxdata, SLI4_BMBX_SIZE)) { 7868 rc = ocs_hw_command(hw, mbxdata, OCS_CMD_NOWAIT, ocs_hw_get_nvparms_cb, cb_arg); 7869 } 7870 7871 if (rc != OCS_HW_RTN_SUCCESS) { 7872 ocs_log_test(hw->os, "READ_NVPARMS failed\n"); 7873 ocs_free(hw->os, mbxdata, SLI4_BMBX_SIZE); 7874 ocs_free(hw->os, cb_arg, sizeof(ocs_hw_get_nvparms_cb_arg_t)); 7875 } 7876 7877 return rc; 7878 } 7879 7880 typedef struct ocs_hw_set_nvparms_cb_arg_s { 7881 ocs_set_nvparms_cb_t cb; 7882 void *arg; 7883 } ocs_hw_set_nvparms_cb_arg_t; 7884 7885 /** 7886 * @brief Called for the completion of set_nvparms for a 7887 * user request. 7888 * 7889 * @param hw Hardware context. 7890 * @param status The status from the MQE. 7891 * @param mqe Pointer to mailbox command buffer. 7892 * @param arg Pointer to a callback argument. 7893 * 7894 * @return Returns 0 on success, or a non-zero value on failure. 7895 */ 7896 static int32_t 7897 ocs_hw_set_nvparms_cb(ocs_hw_t *hw, int32_t status, uint8_t *mqe, void *arg) 7898 { 7899 ocs_hw_set_nvparms_cb_arg_t *cb_arg = arg; 7900 7901 if (cb_arg->cb) { 7902 cb_arg->cb(status, cb_arg->arg); 7903 } 7904 7905 ocs_free(hw->os, mqe, SLI4_BMBX_SIZE); 7906 ocs_free(hw->os, cb_arg, sizeof(ocs_hw_set_nvparms_cb_arg_t)); 7907 7908 return 0; 7909 } 7910 7911 /** 7912 * @ingroup io 7913 * @brief Write non-volatile parms. 7914 * @par Description 7915 * Issues a SLI-4 WRITE_NVPARMS mailbox. When the 7916 * command completes the provided mgmt callback function is 7917 * called. 7918 * 7919 * @param hw Hardware context. 7920 * @param cb Callback function to be called when the 7921 * command completes. 7922 * @param wwpn Port's WWPN in big-endian order, or NULL to use default. 7923 * @param wwnn Port's WWNN in big-endian order, or NULL to use default. 7924 * @param hard_alpa A hard AL_PA address setting used during loop 7925 * initialization. If no hard AL_PA is required, set to 0. 7926 * @param preferred_d_id A preferred D_ID address setting 7927 * that may be overridden with the CONFIG_LINK mailbox command. 7928 * If there is no preference, set to 0. 7929 * @param ul_arg An argument that is passed to the callback 7930 * function. 7931 * 7932 * @return 7933 * - OCS_HW_RTN_SUCCESS on success. 7934 * - OCS_HW_RTN_NO_MEMORY if a malloc fails. 7935 * - OCS_HW_RTN_NO_RESOURCES if unable to get a command 7936 * context. 7937 * - OCS_HW_RTN_ERROR on any other error. 7938 */ 7939 int32_t 7940 ocs_hw_set_nvparms(ocs_hw_t *hw, ocs_set_nvparms_cb_t cb, uint8_t *wwpn, 7941 uint8_t *wwnn, uint8_t hard_alpa, uint32_t preferred_d_id, void* ul_arg) 7942 { 7943 uint8_t *mbxdata; 7944 ocs_hw_set_nvparms_cb_arg_t *cb_arg; 7945 ocs_hw_rtn_e rc = OCS_HW_RTN_SUCCESS; 7946 7947 /* mbxdata holds the header of the command */ 7948 mbxdata = ocs_malloc(hw->os, SLI4_BMBX_SIZE, OCS_M_ZERO | OCS_M_NOWAIT); 7949 if (mbxdata == NULL) { 7950 ocs_log_err(hw->os, "failed to malloc mbox\n"); 7951 return OCS_HW_RTN_NO_MEMORY; 7952 } 7953 7954 /* cb_arg holds the data that will be passed to the callback on completion */ 7955 cb_arg = ocs_malloc(hw->os, sizeof(ocs_hw_set_nvparms_cb_arg_t), OCS_M_NOWAIT); 7956 if (cb_arg == NULL) { 7957 ocs_log_err(hw->os, "failed to malloc cb_arg\n"); 7958 ocs_free(hw->os, mbxdata, SLI4_BMBX_SIZE); 7959 return OCS_HW_RTN_NO_MEMORY; 7960 } 7961 7962 cb_arg->cb = cb; 7963 cb_arg->arg = ul_arg; 7964 7965 if (sli_cmd_write_nvparms(&hw->sli, mbxdata, SLI4_BMBX_SIZE, wwpn, wwnn, hard_alpa, preferred_d_id)) { 7966 rc = ocs_hw_command(hw, mbxdata, OCS_CMD_NOWAIT, ocs_hw_set_nvparms_cb, cb_arg); 7967 } 7968 7969 if (rc != OCS_HW_RTN_SUCCESS) { 7970 ocs_log_test(hw->os, "SET_NVPARMS failed\n"); 7971 ocs_free(hw->os, mbxdata, SLI4_BMBX_SIZE); 7972 ocs_free(hw->os, cb_arg, sizeof(ocs_hw_set_nvparms_cb_arg_t)); 7973 } 7974 7975 return rc; 7976 } 7977 7978 /** 7979 * @brief Called to obtain the count for the specified type. 7980 * 7981 * @param hw Hardware context. 7982 * @param io_count_type IO count type (inuse, free, wait_free). 7983 * 7984 * @return Returns the number of IOs on the specified list type. 7985 */ 7986 uint32_t 7987 ocs_hw_io_get_count(ocs_hw_t *hw, ocs_hw_io_count_type_e io_count_type) 7988 { 7989 ocs_hw_io_t *io = NULL; 7990 uint32_t count = 0; 7991 7992 ocs_lock(&hw->io_lock); 7993 7994 switch (io_count_type) { 7995 case OCS_HW_IO_INUSE_COUNT : 7996 ocs_list_foreach(&hw->io_inuse, io) { 7997 count++; 7998 } 7999 break; 8000 case OCS_HW_IO_FREE_COUNT : 8001 ocs_list_foreach(&hw->io_free, io) { 8002 count++; 8003 } 8004 break; 8005 case OCS_HW_IO_WAIT_FREE_COUNT : 8006 ocs_list_foreach(&hw->io_wait_free, io) { 8007 count++; 8008 } 8009 break; 8010 case OCS_HW_IO_PORT_OWNED_COUNT: 8011 ocs_list_foreach(&hw->io_port_owned, io) { 8012 count++; 8013 } 8014 break; 8015 case OCS_HW_IO_N_TOTAL_IO_COUNT : 8016 count = hw->config.n_io; 8017 break; 8018 } 8019 8020 ocs_unlock(&hw->io_lock); 8021 8022 return count; 8023 } 8024 8025 /** 8026 * @brief Called to obtain the count of produced RQs. 8027 * 8028 * @param hw Hardware context. 8029 * 8030 * @return Returns the number of RQs produced. 8031 */ 8032 uint32_t 8033 ocs_hw_get_rqes_produced_count(ocs_hw_t *hw) 8034 { 8035 uint32_t count = 0; 8036 uint32_t i; 8037 uint32_t j; 8038 8039 for (i = 0; i < hw->hw_rq_count; i++) { 8040 hw_rq_t *rq = hw->hw_rq[i]; 8041 if (rq->rq_tracker != NULL) { 8042 for (j = 0; j < rq->entry_count; j++) { 8043 if (rq->rq_tracker[j] != NULL) { 8044 count++; 8045 } 8046 } 8047 } 8048 } 8049 8050 return count; 8051 } 8052 8053 typedef struct ocs_hw_set_active_profile_cb_arg_s { 8054 ocs_set_active_profile_cb_t cb; 8055 void *arg; 8056 } ocs_hw_set_active_profile_cb_arg_t; 8057 8058 /** 8059 * @brief Called for the completion of set_active_profile for a 8060 * user request. 8061 * 8062 * @param hw Hardware context. 8063 * @param status The status from the MQE 8064 * @param mqe Pointer to mailbox command buffer. 8065 * @param arg Pointer to a callback argument. 8066 * 8067 * @return Returns 0 on success, or a non-zero value on failure. 8068 */ 8069 static int32_t 8070 ocs_hw_set_active_profile_cb(ocs_hw_t *hw, int32_t status, uint8_t *mqe, void *arg) 8071 { 8072 ocs_hw_set_active_profile_cb_arg_t *cb_arg = arg; 8073 8074 if (cb_arg->cb) { 8075 cb_arg->cb(status, cb_arg->arg); 8076 } 8077 8078 ocs_free(hw->os, mqe, SLI4_BMBX_SIZE); 8079 ocs_free(hw->os, cb_arg, sizeof(ocs_hw_get_active_profile_cb_arg_t)); 8080 8081 return 0; 8082 } 8083 8084 /** 8085 * @ingroup io 8086 * @brief Set the currently active profile. 8087 * @par Description 8088 * Issues a SLI4 COMMON_GET_ACTIVE_PROFILE mailbox. When the 8089 * command completes the provided mgmt callback function is 8090 * called. 8091 * 8092 * @param hw Hardware context. 8093 * @param profile_id Profile ID to activate. 8094 * @param cb Callback function to be called when the command completes. 8095 * @param ul_arg An argument that is passed to the callback function. 8096 * 8097 * @return 8098 * - OCS_HW_RTN_SUCCESS on success. 8099 * - OCS_HW_RTN_NO_MEMORY if a malloc fails. 8100 * - OCS_HW_RTN_NO_RESOURCES if unable to get a command 8101 * context. 8102 * - OCS_HW_RTN_ERROR on any other error. 8103 */ 8104 int32_t 8105 ocs_hw_set_active_profile(ocs_hw_t *hw, ocs_set_active_profile_cb_t cb, uint32_t profile_id, void* ul_arg) 8106 { 8107 uint8_t *mbxdata; 8108 ocs_hw_set_active_profile_cb_arg_t *cb_arg; 8109 ocs_hw_rtn_e rc = OCS_HW_RTN_SUCCESS; 8110 8111 /* Only supported on Skyhawk */ 8112 if (sli_get_if_type(&hw->sli) != SLI4_IF_TYPE_BE3_SKH_PF) { 8113 return OCS_HW_RTN_ERROR; 8114 } 8115 8116 /* mbxdata holds the header of the command */ 8117 mbxdata = ocs_malloc(hw->os, SLI4_BMBX_SIZE, OCS_M_ZERO | OCS_M_NOWAIT); 8118 if (mbxdata == NULL) { 8119 ocs_log_err(hw->os, "failed to malloc mbox\n"); 8120 return OCS_HW_RTN_NO_MEMORY; 8121 } 8122 8123 /* cb_arg holds the data that will be passed to the callback on completion */ 8124 cb_arg = ocs_malloc(hw->os, sizeof(ocs_hw_set_active_profile_cb_arg_t), OCS_M_NOWAIT); 8125 if (cb_arg == NULL) { 8126 ocs_log_err(hw->os, "failed to malloc cb_arg\n"); 8127 ocs_free(hw->os, mbxdata, SLI4_BMBX_SIZE); 8128 return OCS_HW_RTN_NO_MEMORY; 8129 } 8130 8131 cb_arg->cb = cb; 8132 cb_arg->arg = ul_arg; 8133 8134 if (sli_cmd_common_set_active_profile(&hw->sli, mbxdata, SLI4_BMBX_SIZE, 0, profile_id)) { 8135 rc = ocs_hw_command(hw, mbxdata, OCS_CMD_NOWAIT, ocs_hw_set_active_profile_cb, cb_arg); 8136 } 8137 8138 if (rc != OCS_HW_RTN_SUCCESS) { 8139 ocs_log_test(hw->os, "SET_ACTIVE_PROFILE failed\n"); 8140 ocs_free(hw->os, mbxdata, SLI4_BMBX_SIZE); 8141 ocs_free(hw->os, cb_arg, sizeof(ocs_hw_set_active_profile_cb_arg_t)); 8142 } 8143 8144 return rc; 8145 } 8146 8147 /* 8148 * Private functions 8149 */ 8150 8151 /** 8152 * @brief Update the queue hash with the ID and index. 8153 * 8154 * @param hash Pointer to hash table. 8155 * @param id ID that was created. 8156 * @param index The index into the hash object. 8157 */ 8158 static void 8159 ocs_hw_queue_hash_add(ocs_queue_hash_t *hash, uint16_t id, uint16_t index) 8160 { 8161 uint32_t hash_index = id & (OCS_HW_Q_HASH_SIZE - 1); 8162 8163 /* 8164 * Since the hash is always bigger than the number of queues, then we 8165 * never have to worry about an infinite loop. 8166 */ 8167 while(hash[hash_index].in_use) { 8168 hash_index = (hash_index + 1) & (OCS_HW_Q_HASH_SIZE - 1); 8169 } 8170 8171 /* not used, claim the entry */ 8172 hash[hash_index].id = id; 8173 hash[hash_index].in_use = 1; 8174 hash[hash_index].index = index; 8175 } 8176 8177 /** 8178 * @brief Find index given queue ID. 8179 * 8180 * @param hash Pointer to hash table. 8181 * @param id ID to find. 8182 * 8183 * @return Returns the index into the HW cq array or -1 if not found. 8184 */ 8185 int32_t 8186 ocs_hw_queue_hash_find(ocs_queue_hash_t *hash, uint16_t id) 8187 { 8188 int32_t rc = -1; 8189 int32_t index = id & (OCS_HW_Q_HASH_SIZE - 1); 8190 8191 /* 8192 * Since the hash is always bigger than the maximum number of Qs, then we 8193 * never have to worry about an infinite loop. We will always find an 8194 * unused entry. 8195 */ 8196 do { 8197 if (hash[index].in_use && 8198 hash[index].id == id) { 8199 rc = hash[index].index; 8200 } else { 8201 index = (index + 1) & (OCS_HW_Q_HASH_SIZE - 1); 8202 } 8203 } while(rc == -1 && hash[index].in_use); 8204 8205 return rc; 8206 } 8207 8208 static int32_t 8209 ocs_hw_domain_add(ocs_hw_t *hw, ocs_domain_t *domain) 8210 { 8211 int32_t rc = OCS_HW_RTN_ERROR; 8212 uint16_t fcfi = UINT16_MAX; 8213 8214 if ((hw == NULL) || (domain == NULL)) { 8215 ocs_log_err(NULL, "bad parameter hw=%p domain=%p\n", 8216 hw, domain); 8217 return OCS_HW_RTN_ERROR; 8218 } 8219 8220 fcfi = domain->fcf_indicator; 8221 8222 if (fcfi < SLI4_MAX_FCFI) { 8223 uint16_t fcf_index = UINT16_MAX; 8224 8225 ocs_log_debug(hw->os, "adding domain %p @ %#x\n", 8226 domain, fcfi); 8227 hw->domains[fcfi] = domain; 8228 8229 /* HW_WORKAROUND_OVERRIDE_FCFI_IN_SRB */ 8230 if (hw->workaround.override_fcfi) { 8231 if (hw->first_domain_idx < 0) { 8232 hw->first_domain_idx = fcfi; 8233 } 8234 } 8235 8236 fcf_index = domain->fcf; 8237 8238 if (fcf_index < SLI4_MAX_FCF_INDEX) { 8239 ocs_log_debug(hw->os, "adding map of FCF index %d to FCFI %d\n", 8240 fcf_index, fcfi); 8241 hw->fcf_index_fcfi[fcf_index] = fcfi; 8242 rc = OCS_HW_RTN_SUCCESS; 8243 } else { 8244 ocs_log_test(hw->os, "FCF index %d out of range (max %d)\n", 8245 fcf_index, SLI4_MAX_FCF_INDEX); 8246 hw->domains[fcfi] = NULL; 8247 } 8248 } else { 8249 ocs_log_test(hw->os, "FCFI %#x out of range (max %#x)\n", 8250 fcfi, SLI4_MAX_FCFI); 8251 } 8252 8253 return rc; 8254 } 8255 8256 static int32_t 8257 ocs_hw_domain_del(ocs_hw_t *hw, ocs_domain_t *domain) 8258 { 8259 int32_t rc = OCS_HW_RTN_ERROR; 8260 uint16_t fcfi = UINT16_MAX; 8261 8262 if ((hw == NULL) || (domain == NULL)) { 8263 ocs_log_err(NULL, "bad parameter hw=%p domain=%p\n", 8264 hw, domain); 8265 return OCS_HW_RTN_ERROR; 8266 } 8267 8268 fcfi = domain->fcf_indicator; 8269 8270 if (fcfi < SLI4_MAX_FCFI) { 8271 uint16_t fcf_index = UINT16_MAX; 8272 8273 ocs_log_debug(hw->os, "deleting domain %p @ %#x\n", 8274 domain, fcfi); 8275 8276 if (domain != hw->domains[fcfi]) { 8277 ocs_log_test(hw->os, "provided domain %p does not match stored domain %p\n", 8278 domain, hw->domains[fcfi]); 8279 return OCS_HW_RTN_ERROR; 8280 } 8281 8282 hw->domains[fcfi] = NULL; 8283 8284 /* HW_WORKAROUND_OVERRIDE_FCFI_IN_SRB */ 8285 if (hw->workaround.override_fcfi) { 8286 if (hw->first_domain_idx == fcfi) { 8287 hw->first_domain_idx = -1; 8288 } 8289 } 8290 8291 fcf_index = domain->fcf; 8292 8293 if (fcf_index < SLI4_MAX_FCF_INDEX) { 8294 if (hw->fcf_index_fcfi[fcf_index] == fcfi) { 8295 hw->fcf_index_fcfi[fcf_index] = 0; 8296 rc = OCS_HW_RTN_SUCCESS; 8297 } else { 8298 ocs_log_test(hw->os, "indexed FCFI %#x doesn't match provided %#x @ %d\n", 8299 hw->fcf_index_fcfi[fcf_index], fcfi, fcf_index); 8300 } 8301 } else { 8302 ocs_log_test(hw->os, "FCF index %d out of range (max %d)\n", 8303 fcf_index, SLI4_MAX_FCF_INDEX); 8304 } 8305 } else { 8306 ocs_log_test(hw->os, "FCFI %#x out of range (max %#x)\n", 8307 fcfi, SLI4_MAX_FCFI); 8308 } 8309 8310 return rc; 8311 } 8312 8313 ocs_domain_t * 8314 ocs_hw_domain_get(ocs_hw_t *hw, uint16_t fcfi) 8315 { 8316 8317 if (hw == NULL) { 8318 ocs_log_err(NULL, "bad parameter hw=%p\n", hw); 8319 return NULL; 8320 } 8321 8322 if (fcfi < SLI4_MAX_FCFI) { 8323 return hw->domains[fcfi]; 8324 } else { 8325 ocs_log_test(hw->os, "FCFI %#x out of range (max %#x)\n", 8326 fcfi, SLI4_MAX_FCFI); 8327 return NULL; 8328 } 8329 } 8330 8331 static ocs_domain_t * 8332 ocs_hw_domain_get_indexed(ocs_hw_t *hw, uint16_t fcf_index) 8333 { 8334 8335 if (hw == NULL) { 8336 ocs_log_err(NULL, "bad parameter hw=%p\n", hw); 8337 return NULL; 8338 } 8339 8340 if (fcf_index < SLI4_MAX_FCF_INDEX) { 8341 return ocs_hw_domain_get(hw, hw->fcf_index_fcfi[fcf_index]); 8342 } else { 8343 ocs_log_test(hw->os, "FCF index %d out of range (max %d)\n", 8344 fcf_index, SLI4_MAX_FCF_INDEX); 8345 return NULL; 8346 } 8347 } 8348 8349 /** 8350 * @brief Quaratine an IO by taking a reference count and adding it to the 8351 * quarantine list. When the IO is popped from the list then the 8352 * count is released and the IO MAY be freed depending on whether 8353 * it is still referenced by the IO. 8354 * 8355 * @n @b Note: BZ 160124 - If this is a target write or an initiator read using 8356 * DIF, then we must add the XRI to a quarantine list until we receive 8357 * 4 more completions of this same type. 8358 * 8359 * @param hw Hardware context. 8360 * @param wq Pointer to the WQ associated with the IO object to quarantine. 8361 * @param io Pointer to the io object to quarantine. 8362 */ 8363 static void 8364 ocs_hw_io_quarantine(ocs_hw_t *hw, hw_wq_t *wq, ocs_hw_io_t *io) 8365 { 8366 ocs_quarantine_info_t *q_info = &wq->quarantine_info; 8367 uint32_t index; 8368 ocs_hw_io_t *free_io = NULL; 8369 8370 /* return if the QX bit was clear */ 8371 if (!io->quarantine) { 8372 return; 8373 } 8374 8375 /* increment the IO refcount to prevent it from being freed before the quarantine is over */ 8376 if (ocs_ref_get_unless_zero(&io->ref) == 0) { 8377 /* command no longer active */ 8378 ocs_log_debug(hw ? hw->os : NULL, 8379 "io not active xri=0x%x tag=0x%x\n", 8380 io->indicator, io->reqtag); 8381 return; 8382 } 8383 8384 sli_queue_lock(wq->queue); 8385 index = q_info->quarantine_index; 8386 free_io = q_info->quarantine_ios[index]; 8387 q_info->quarantine_ios[index] = io; 8388 q_info->quarantine_index = (index + 1) % OCS_HW_QUARANTINE_QUEUE_DEPTH; 8389 sli_queue_unlock(wq->queue); 8390 8391 if (free_io != NULL) { 8392 ocs_ref_put(&free_io->ref); /* ocs_ref_get(): same function */ 8393 } 8394 } 8395 8396 /** 8397 * @brief Process entries on the given completion queue. 8398 * 8399 * @param hw Hardware context. 8400 * @param cq Pointer to the HW completion queue object. 8401 * 8402 * @return None. 8403 */ 8404 void 8405 ocs_hw_cq_process(ocs_hw_t *hw, hw_cq_t *cq) 8406 { 8407 uint8_t cqe[sizeof(sli4_mcqe_t)]; 8408 uint16_t rid = UINT16_MAX; 8409 sli4_qentry_e ctype; /* completion type */ 8410 int32_t status; 8411 uint32_t n_processed = 0; 8412 time_t tstart; 8413 time_t telapsed; 8414 8415 tstart = ocs_msectime(); 8416 8417 while (!sli_queue_read(&hw->sli, cq->queue, cqe)) { 8418 status = sli_cq_parse(&hw->sli, cq->queue, cqe, &ctype, &rid); 8419 /* 8420 * The sign of status is significant. If status is: 8421 * == 0 : call completed correctly and the CQE indicated success 8422 * > 0 : call completed correctly and the CQE indicated an error 8423 * < 0 : call failed and no information is available about the CQE 8424 */ 8425 if (status < 0) { 8426 if (status == -2) { 8427 /* Notification that an entry was consumed, but not completed */ 8428 continue; 8429 } 8430 8431 break; 8432 } 8433 8434 switch (ctype) { 8435 case SLI_QENTRY_ASYNC: 8436 CPUTRACE("async"); 8437 sli_cqe_async(&hw->sli, cqe); 8438 break; 8439 case SLI_QENTRY_MQ: 8440 /* 8441 * Process MQ entry. Note there is no way to determine 8442 * the MQ_ID from the completion entry. 8443 */ 8444 CPUTRACE("mq"); 8445 ocs_hw_mq_process(hw, status, hw->mq); 8446 break; 8447 case SLI_QENTRY_OPT_WRITE_CMD: 8448 ocs_hw_rqpair_process_auto_xfr_rdy_cmd(hw, cq, cqe); 8449 break; 8450 case SLI_QENTRY_OPT_WRITE_DATA: 8451 ocs_hw_rqpair_process_auto_xfr_rdy_data(hw, cq, cqe); 8452 break; 8453 case SLI_QENTRY_WQ: 8454 CPUTRACE("wq"); 8455 ocs_hw_wq_process(hw, cq, cqe, status, rid); 8456 break; 8457 case SLI_QENTRY_WQ_RELEASE: { 8458 uint32_t wq_id = rid; 8459 int32_t index = ocs_hw_queue_hash_find(hw->wq_hash, wq_id); 8460 8461 if (unlikely(index < 0)) { 8462 ocs_log_err(hw->os, "unknown idx=%#x rid=%#x\n", 8463 index, rid); 8464 break; 8465 } 8466 8467 hw_wq_t *wq = hw->hw_wq[index]; 8468 8469 /* Submit any HW IOs that are on the WQ pending list */ 8470 hw_wq_submit_pending(wq, wq->wqec_set_count); 8471 8472 break; 8473 } 8474 8475 case SLI_QENTRY_RQ: 8476 CPUTRACE("rq"); 8477 ocs_hw_rqpair_process_rq(hw, cq, cqe); 8478 break; 8479 case SLI_QENTRY_XABT: { 8480 CPUTRACE("xabt"); 8481 ocs_hw_xabt_process(hw, cq, cqe, rid); 8482 break; 8483 } 8484 default: 8485 ocs_log_test(hw->os, "unhandled ctype=%#x rid=%#x\n", ctype, rid); 8486 break; 8487 } 8488 8489 n_processed++; 8490 if (n_processed == cq->queue->proc_limit) { 8491 break; 8492 } 8493 8494 if (cq->queue->n_posted >= (cq->queue->posted_limit)) { 8495 sli_queue_arm(&hw->sli, cq->queue, FALSE); 8496 } 8497 } 8498 8499 sli_queue_arm(&hw->sli, cq->queue, TRUE); 8500 8501 if (n_processed > cq->queue->max_num_processed) { 8502 cq->queue->max_num_processed = n_processed; 8503 } 8504 telapsed = ocs_msectime() - tstart; 8505 if (telapsed > cq->queue->max_process_time) { 8506 cq->queue->max_process_time = telapsed; 8507 } 8508 } 8509 8510 /** 8511 * @brief Process WQ completion queue entries. 8512 * 8513 * @param hw Hardware context. 8514 * @param cq Pointer to the HW completion queue object. 8515 * @param cqe Pointer to WQ completion queue. 8516 * @param status Completion status. 8517 * @param rid Resource ID (IO tag). 8518 * 8519 * @return none 8520 */ 8521 void 8522 ocs_hw_wq_process(ocs_hw_t *hw, hw_cq_t *cq, uint8_t *cqe, int32_t status, uint16_t rid) 8523 { 8524 hw_wq_callback_t *wqcb; 8525 8526 ocs_queue_history_cqe(&hw->q_hist, SLI_QENTRY_WQ, (void *)cqe, ((sli4_fc_wcqe_t *)cqe)->status, cq->queue->id, 8527 ((cq->queue->index - 1) & (cq->queue->length - 1))); 8528 8529 if(rid == OCS_HW_REQUE_XRI_REGTAG) { 8530 if(status) { 8531 ocs_log_err(hw->os, "reque xri failed, status = %d \n", status); 8532 } 8533 return; 8534 } 8535 8536 wqcb = ocs_hw_reqtag_get_instance(hw, rid); 8537 if (wqcb == NULL) { 8538 ocs_log_err(hw->os, "invalid request tag: x%x\n", rid); 8539 return; 8540 } 8541 8542 if (wqcb->callback == NULL) { 8543 ocs_log_err(hw->os, "wqcb callback is NULL\n"); 8544 return; 8545 } 8546 8547 (*wqcb->callback)(wqcb->arg, cqe, status); 8548 } 8549 8550 /** 8551 * @brief Process WQ completions for IO requests 8552 * 8553 * @param arg Generic callback argument 8554 * @param cqe Pointer to completion queue entry 8555 * @param status Completion status 8556 * 8557 * @par Description 8558 * @n @b Note: Regarding io->reqtag, the reqtag is assigned once when HW IOs are initialized 8559 * in ocs_hw_setup_io(), and don't need to be returned to the hw->wq_reqtag_pool. 8560 * 8561 * @return None. 8562 */ 8563 static void 8564 ocs_hw_wq_process_io(void *arg, uint8_t *cqe, int32_t status) 8565 { 8566 ocs_hw_io_t *io = arg; 8567 ocs_hw_t *hw = io->hw; 8568 sli4_fc_wcqe_t *wcqe = (void *)cqe; 8569 uint32_t len = 0; 8570 uint32_t ext = 0; 8571 uint8_t out_of_order_axr_cmd = 0; 8572 uint8_t out_of_order_axr_data = 0; 8573 uint8_t lock_taken = 0; 8574 #if defined(OCS_DISC_SPIN_DELAY) 8575 uint32_t delay = 0; 8576 char prop_buf[32]; 8577 #endif 8578 8579 /* 8580 * For the primary IO, this will also be used for the 8581 * response. So it is important to only set/clear this 8582 * flag on the first data phase of the IO because 8583 * subsequent phases will be done on the secondary XRI. 8584 */ 8585 if (io->quarantine && io->quarantine_first_phase) { 8586 io->quarantine = (wcqe->qx == 1); 8587 ocs_hw_io_quarantine(hw, io->wq, io); 8588 } 8589 io->quarantine_first_phase = FALSE; 8590 8591 /* BZ 161832 - free secondary HW IO */ 8592 if (io->sec_hio != NULL && 8593 io->sec_hio->quarantine) { 8594 /* 8595 * If the quarantine flag is set on the 8596 * IO, then set it on the secondary IO 8597 * based on the quarantine XRI (QX) bit 8598 * sent by the FW. 8599 */ 8600 io->sec_hio->quarantine = (wcqe->qx == 1); 8601 /* use the primary io->wq because it is not set on the secondary IO. */ 8602 ocs_hw_io_quarantine(hw, io->wq, io->sec_hio); 8603 } 8604 8605 ocs_hw_remove_io_timed_wqe(hw, io); 8606 8607 /* clear xbusy flag if WCQE[XB] is clear */ 8608 if (io->xbusy && wcqe->xb == 0) { 8609 io->xbusy = FALSE; 8610 } 8611 8612 /* get extended CQE status */ 8613 switch (io->type) { 8614 case OCS_HW_BLS_ACC: 8615 case OCS_HW_BLS_ACC_SID: 8616 break; 8617 case OCS_HW_ELS_REQ: 8618 sli_fc_els_did(&hw->sli, cqe, &ext); 8619 len = sli_fc_response_length(&hw->sli, cqe); 8620 break; 8621 case OCS_HW_ELS_RSP: 8622 case OCS_HW_ELS_RSP_SID: 8623 case OCS_HW_FC_CT_RSP: 8624 break; 8625 case OCS_HW_FC_CT: 8626 len = sli_fc_response_length(&hw->sli, cqe); 8627 break; 8628 case OCS_HW_IO_TARGET_WRITE: 8629 len = sli_fc_io_length(&hw->sli, cqe); 8630 #if defined(OCS_DISC_SPIN_DELAY) 8631 if (ocs_get_property("disk_spin_delay", prop_buf, sizeof(prop_buf)) == 0) { 8632 delay = ocs_strtoul(prop_buf, 0, 0); 8633 ocs_udelay(delay); 8634 } 8635 #endif 8636 break; 8637 case OCS_HW_IO_TARGET_READ: 8638 len = sli_fc_io_length(&hw->sli, cqe); 8639 /* 8640 * if_type == 2 seems to return 0 "total length placed" on 8641 * FCP_TSEND64_WQE completions. If this appears to happen, 8642 * use the CTIO data transfer length instead. 8643 */ 8644 if (hw->workaround.retain_tsend_io_length && !len && !status) { 8645 len = io->length; 8646 } 8647 8648 break; 8649 case OCS_HW_IO_TARGET_RSP: 8650 if(io->is_port_owned) { 8651 ocs_lock(&io->axr_lock); 8652 lock_taken = 1; 8653 if(io->axr_buf->call_axr_cmd) { 8654 out_of_order_axr_cmd = 1; 8655 } 8656 if(io->axr_buf->call_axr_data) { 8657 out_of_order_axr_data = 1; 8658 } 8659 } 8660 break; 8661 case OCS_HW_IO_INITIATOR_READ: 8662 len = sli_fc_io_length(&hw->sli, cqe); 8663 break; 8664 case OCS_HW_IO_INITIATOR_WRITE: 8665 len = sli_fc_io_length(&hw->sli, cqe); 8666 break; 8667 case OCS_HW_IO_INITIATOR_NODATA: 8668 break; 8669 case OCS_HW_IO_DNRX_REQUEUE: 8670 /* release the count for re-posting the buffer */ 8671 //ocs_hw_io_free(hw, io); 8672 break; 8673 default: 8674 ocs_log_test(hw->os, "XXX unhandled io type %#x for XRI 0x%x\n", 8675 io->type, io->indicator); 8676 break; 8677 } 8678 if (status) { 8679 ext = sli_fc_ext_status(&hw->sli, cqe); 8680 /* Emulate IAAB=0 for initiator WQEs only; i.e. automatically 8681 * abort exchange if an error occurred and exchange is still busy. 8682 */ 8683 if (hw->config.i_only_aab && 8684 (ocs_hw_iotype_is_originator(io->type)) && 8685 (ocs_hw_wcqe_abort_needed(status, ext, wcqe->xb))) { 8686 ocs_hw_rtn_e rc; 8687 8688 ocs_log_debug(hw->os, "aborting xri=%#x tag=%#x\n", 8689 io->indicator, io->reqtag); 8690 /* 8691 * Because the initiator will not issue another IO phase, then it is OK to to issue the 8692 * callback on the abort completion, but for consistency with the target, wait for the 8693 * XRI_ABORTED CQE to issue the IO callback. 8694 */ 8695 rc = ocs_hw_io_abort(hw, io, TRUE, NULL, NULL); 8696 8697 if (rc == OCS_HW_RTN_SUCCESS) { 8698 /* latch status to return after abort is complete */ 8699 io->status_saved = 1; 8700 io->saved_status = status; 8701 io->saved_ext = ext; 8702 io->saved_len = len; 8703 goto exit_ocs_hw_wq_process_io; 8704 } else if (rc == OCS_HW_RTN_IO_ABORT_IN_PROGRESS) { 8705 /* 8706 * Already being aborted by someone else (ABTS 8707 * perhaps). Just fall through and return original 8708 * error. 8709 */ 8710 ocs_log_debug(hw->os, "abort in progress xri=%#x tag=%#x\n", 8711 io->indicator, io->reqtag); 8712 8713 } else { 8714 /* Failed to abort for some other reason, log error */ 8715 ocs_log_test(hw->os, "Failed to abort xri=%#x tag=%#x rc=%d\n", 8716 io->indicator, io->reqtag, rc); 8717 } 8718 } 8719 8720 /* 8721 * If we're not an originator IO, and XB is set, then issue abort for the IO from within the HW 8722 */ 8723 if ( (! ocs_hw_iotype_is_originator(io->type)) && wcqe->xb) { 8724 ocs_hw_rtn_e rc; 8725 8726 ocs_log_debug(hw->os, "aborting xri=%#x tag=%#x\n", io->indicator, io->reqtag); 8727 8728 /* 8729 * Because targets may send a response when the IO completes using the same XRI, we must 8730 * wait for the XRI_ABORTED CQE to issue the IO callback 8731 */ 8732 rc = ocs_hw_io_abort(hw, io, FALSE, NULL, NULL); 8733 if (rc == OCS_HW_RTN_SUCCESS) { 8734 /* latch status to return after abort is complete */ 8735 io->status_saved = 1; 8736 io->saved_status = status; 8737 io->saved_ext = ext; 8738 io->saved_len = len; 8739 goto exit_ocs_hw_wq_process_io; 8740 } else if (rc == OCS_HW_RTN_IO_ABORT_IN_PROGRESS) { 8741 /* 8742 * Already being aborted by someone else (ABTS 8743 * perhaps). Just fall through and return original 8744 * error. 8745 */ 8746 ocs_log_debug(hw->os, "abort in progress xri=%#x tag=%#x\n", 8747 io->indicator, io->reqtag); 8748 8749 } else { 8750 /* Failed to abort for some other reason, log error */ 8751 ocs_log_test(hw->os, "Failed to abort xri=%#x tag=%#x rc=%d\n", 8752 io->indicator, io->reqtag, rc); 8753 } 8754 } 8755 } 8756 /* BZ 161832 - free secondary HW IO */ 8757 if (io->sec_hio != NULL) { 8758 ocs_hw_io_free(hw, io->sec_hio); 8759 io->sec_hio = NULL; 8760 } 8761 8762 if (io->done != NULL) { 8763 ocs_hw_done_t done = io->done; 8764 void *arg = io->arg; 8765 8766 io->done = NULL; 8767 8768 if (io->status_saved) { 8769 /* use latched status if exists */ 8770 status = io->saved_status; 8771 len = io->saved_len; 8772 ext = io->saved_ext; 8773 io->status_saved = 0; 8774 } 8775 8776 /* Restore default SGL */ 8777 ocs_hw_io_restore_sgl(hw, io); 8778 done(io, io->rnode, len, status, ext, arg); 8779 } 8780 8781 if(out_of_order_axr_cmd) { 8782 /* bounce enabled, single RQ, we snoop the ox_id to choose the cpuidx */ 8783 if (hw->config.bounce) { 8784 fc_header_t *hdr = io->axr_buf->cmd_seq->header->dma.virt; 8785 uint32_t s_id = fc_be24toh(hdr->s_id); 8786 uint32_t d_id = fc_be24toh(hdr->d_id); 8787 uint32_t ox_id = ocs_be16toh(hdr->ox_id); 8788 if (hw->callback.bounce != NULL) { 8789 (*hw->callback.bounce)(ocs_hw_unsol_process_bounce, io->axr_buf->cmd_seq, s_id, d_id, ox_id); 8790 } 8791 }else { 8792 hw->callback.unsolicited(hw->args.unsolicited, io->axr_buf->cmd_seq); 8793 } 8794 8795 if(out_of_order_axr_data) { 8796 /* bounce enabled, single RQ, we snoop the ox_id to choose the cpuidx */ 8797 if (hw->config.bounce) { 8798 fc_header_t *hdr = io->axr_buf->seq.header->dma.virt; 8799 uint32_t s_id = fc_be24toh(hdr->s_id); 8800 uint32_t d_id = fc_be24toh(hdr->d_id); 8801 uint32_t ox_id = ocs_be16toh(hdr->ox_id); 8802 if (hw->callback.bounce != NULL) { 8803 (*hw->callback.bounce)(ocs_hw_unsol_process_bounce, &io->axr_buf->seq, s_id, d_id, ox_id); 8804 } 8805 }else { 8806 hw->callback.unsolicited(hw->args.unsolicited, &io->axr_buf->seq); 8807 } 8808 } 8809 } 8810 8811 exit_ocs_hw_wq_process_io: 8812 if(lock_taken) { 8813 ocs_unlock(&io->axr_lock); 8814 } 8815 } 8816 8817 /** 8818 * @brief Process WQ completions for abort requests. 8819 * 8820 * @param arg Generic callback argument. 8821 * @param cqe Pointer to completion queue entry. 8822 * @param status Completion status. 8823 * 8824 * @return None. 8825 */ 8826 static void 8827 ocs_hw_wq_process_abort(void *arg, uint8_t *cqe, int32_t status) 8828 { 8829 ocs_hw_io_t *io = arg; 8830 ocs_hw_t *hw = io->hw; 8831 uint32_t ext = 0; 8832 uint32_t len = 0; 8833 hw_wq_callback_t *wqcb; 8834 8835 /* 8836 * For IOs that were aborted internally, we may need to issue the callback here depending 8837 * on whether a XRI_ABORTED CQE is expected ot not. If the status is Local Reject/No XRI, then 8838 * issue the callback now. 8839 */ 8840 ext = sli_fc_ext_status(&hw->sli, cqe); 8841 if (status == SLI4_FC_WCQE_STATUS_LOCAL_REJECT && 8842 ext == SLI4_FC_LOCAL_REJECT_NO_XRI && 8843 io->done != NULL) { 8844 ocs_hw_done_t done = io->done; 8845 void *arg = io->arg; 8846 8847 io->done = NULL; 8848 8849 /* 8850 * Use latched status as this is always saved for an internal abort 8851 * 8852 * Note: We wont have both a done and abort_done function, so don't worry about 8853 * clobbering the len, status and ext fields. 8854 */ 8855 status = io->saved_status; 8856 len = io->saved_len; 8857 ext = io->saved_ext; 8858 io->status_saved = 0; 8859 done(io, io->rnode, len, status, ext, arg); 8860 } 8861 8862 if (io->abort_done != NULL) { 8863 ocs_hw_done_t done = io->abort_done; 8864 void *arg = io->abort_arg; 8865 8866 io->abort_done = NULL; 8867 8868 done(io, io->rnode, len, status, ext, arg); 8869 } 8870 ocs_lock(&hw->io_abort_lock); 8871 /* clear abort bit to indicate abort is complete */ 8872 io->abort_in_progress = 0; 8873 ocs_unlock(&hw->io_abort_lock); 8874 8875 /* Free the WQ callback */ 8876 ocs_hw_assert(io->abort_reqtag != UINT32_MAX); 8877 wqcb = ocs_hw_reqtag_get_instance(hw, io->abort_reqtag); 8878 ocs_hw_reqtag_free(hw, wqcb); 8879 8880 /* 8881 * Call ocs_hw_io_free() because this releases the WQ reservation as 8882 * well as doing the refcount put. Don't duplicate the code here. 8883 */ 8884 (void)ocs_hw_io_free(hw, io); 8885 } 8886 8887 /** 8888 * @brief Process XABT completions 8889 * 8890 * @param hw Hardware context. 8891 * @param cq Pointer to the HW completion queue object. 8892 * @param cqe Pointer to WQ completion queue. 8893 * @param rid Resource ID (IO tag). 8894 * 8895 * 8896 * @return None. 8897 */ 8898 void 8899 ocs_hw_xabt_process(ocs_hw_t *hw, hw_cq_t *cq, uint8_t *cqe, uint16_t rid) 8900 { 8901 /* search IOs wait free list */ 8902 ocs_hw_io_t *io = NULL; 8903 8904 io = ocs_hw_io_lookup(hw, rid); 8905 8906 ocs_queue_history_cqe(&hw->q_hist, SLI_QENTRY_XABT, (void *)cqe, 0, cq->queue->id, 8907 ((cq->queue->index - 1) & (cq->queue->length - 1))); 8908 if (io == NULL) { 8909 /* IO lookup failure should never happen */ 8910 ocs_log_err(hw->os, "Error: xabt io lookup failed rid=%#x\n", rid); 8911 return; 8912 } 8913 8914 if (!io->xbusy) { 8915 ocs_log_debug(hw->os, "xabt io not busy rid=%#x\n", rid); 8916 } else { 8917 /* mark IO as no longer busy */ 8918 io->xbusy = FALSE; 8919 } 8920 8921 if (io->is_port_owned) { 8922 ocs_lock(&hw->io_lock); 8923 /* Take reference so that below callback will not free io before reque */ 8924 ocs_ref_get(&io->ref); 8925 ocs_unlock(&hw->io_lock); 8926 } 8927 8928 /* For IOs that were aborted internally, we need to issue any pending callback here. */ 8929 if (io->done != NULL) { 8930 ocs_hw_done_t done = io->done; 8931 void *arg = io->arg; 8932 8933 /* Use latched status as this is always saved for an internal abort */ 8934 int32_t status = io->saved_status; 8935 uint32_t len = io->saved_len; 8936 uint32_t ext = io->saved_ext; 8937 8938 io->done = NULL; 8939 io->status_saved = 0; 8940 8941 done(io, io->rnode, len, status, ext, arg); 8942 } 8943 8944 /* Check to see if this is a port owned XRI */ 8945 if (io->is_port_owned) { 8946 ocs_lock(&hw->io_lock); 8947 ocs_hw_reque_xri(hw, io); 8948 ocs_unlock(&hw->io_lock); 8949 /* Not hanlding reque xri completion, free io */ 8950 ocs_hw_io_free(hw, io); 8951 return; 8952 } 8953 8954 ocs_lock(&hw->io_lock); 8955 if ((io->state == OCS_HW_IO_STATE_INUSE) || (io->state == OCS_HW_IO_STATE_WAIT_FREE)) { 8956 /* if on wait_free list, caller has already freed IO; 8957 * remove from wait_free list and add to free list. 8958 * if on in-use list, already marked as no longer busy; 8959 * just leave there and wait for caller to free. 8960 */ 8961 if (io->state == OCS_HW_IO_STATE_WAIT_FREE) { 8962 io->state = OCS_HW_IO_STATE_FREE; 8963 ocs_list_remove(&hw->io_wait_free, io); 8964 ocs_hw_io_free_move_correct_list(hw, io); 8965 } 8966 } 8967 ocs_unlock(&hw->io_lock); 8968 } 8969 8970 /** 8971 * @brief Adjust the number of WQs and CQs within the HW. 8972 * 8973 * @par Description 8974 * Calculates the number of WQs and associated CQs needed in the HW based on 8975 * the number of IOs. Calculates the starting CQ index for each WQ, RQ and 8976 * MQ. 8977 * 8978 * @param hw Hardware context allocated by the caller. 8979 */ 8980 static void 8981 ocs_hw_adjust_wqs(ocs_hw_t *hw) 8982 { 8983 uint32_t max_wq_num = sli_get_max_queue(&hw->sli, SLI_QTYPE_WQ); 8984 uint32_t max_wq_entries = hw->num_qentries[SLI_QTYPE_WQ]; 8985 uint32_t max_cq_entries = hw->num_qentries[SLI_QTYPE_CQ]; 8986 8987 /* 8988 * possibly adjust the the size of the WQs so that the CQ is twice as 8989 * big as the WQ to allow for 2 completions per IO. This allows us to 8990 * handle multi-phase as well as aborts. 8991 */ 8992 if (max_cq_entries < max_wq_entries * 2) { 8993 max_wq_entries = hw->num_qentries[SLI_QTYPE_WQ] = max_cq_entries / 2; 8994 } 8995 8996 /* 8997 * Calculate the number of WQs to use base on the number of IOs. 8998 * 8999 * Note: We need to reserve room for aborts which must be sent down 9000 * the same WQ as the IO. So we allocate enough WQ space to 9001 * handle 2 times the number of IOs. Half of the space will be 9002 * used for normal IOs and the other hwf is reserved for aborts. 9003 */ 9004 hw->config.n_wq = ((hw->config.n_io * 2) + (max_wq_entries - 1)) / max_wq_entries; 9005 9006 /* 9007 * For performance reasons, it is best to use use a minimum of 4 WQs 9008 * for BE3 and Skyhawk. 9009 */ 9010 if (hw->config.n_wq < 4 && 9011 SLI4_IF_TYPE_BE3_SKH_PF == sli_get_if_type(&hw->sli)) { 9012 hw->config.n_wq = 4; 9013 } 9014 9015 /* 9016 * For dual-chute support, we need to have at least one WQ per chute. 9017 */ 9018 if (hw->config.n_wq < 2 && 9019 ocs_hw_get_num_chutes(hw) > 1) { 9020 hw->config.n_wq = 2; 9021 } 9022 9023 /* make sure we haven't exceeded the max supported in the HW */ 9024 if (hw->config.n_wq > OCS_HW_MAX_NUM_WQ) { 9025 hw->config.n_wq = OCS_HW_MAX_NUM_WQ; 9026 } 9027 9028 /* make sure we haven't exceeded the chip maximum */ 9029 if (hw->config.n_wq > max_wq_num) { 9030 hw->config.n_wq = max_wq_num; 9031 } 9032 9033 /* 9034 * Using Queue Topology string, we divide by number of chutes 9035 */ 9036 hw->config.n_wq /= ocs_hw_get_num_chutes(hw); 9037 } 9038 9039 static int32_t 9040 ocs_hw_command_process(ocs_hw_t *hw, int32_t status, uint8_t *mqe, size_t size) 9041 { 9042 ocs_command_ctx_t *ctx = NULL; 9043 9044 ocs_lock(&hw->cmd_lock); 9045 if (NULL == (ctx = ocs_list_remove_head(&hw->cmd_head))) { 9046 ocs_log_err(hw->os, "XXX no command context?!?\n"); 9047 ocs_unlock(&hw->cmd_lock); 9048 return -1; 9049 } 9050 9051 hw->cmd_head_count--; 9052 9053 /* Post any pending requests */ 9054 ocs_hw_cmd_submit_pending(hw); 9055 9056 ocs_unlock(&hw->cmd_lock); 9057 9058 if (ctx->cb) { 9059 if (ctx->buf) { 9060 ocs_memcpy(ctx->buf, mqe, size); 9061 } 9062 ctx->cb(hw, status, ctx->buf, ctx->arg); 9063 } 9064 9065 ocs_memset(ctx, 0, sizeof(ocs_command_ctx_t)); 9066 ocs_free(hw->os, ctx, sizeof(ocs_command_ctx_t)); 9067 9068 return 0; 9069 } 9070 9071 /** 9072 * @brief Process entries on the given mailbox queue. 9073 * 9074 * @param hw Hardware context. 9075 * @param status CQE status. 9076 * @param mq Pointer to the mailbox queue object. 9077 * 9078 * @return Returns 0 on success, or a non-zero value on failure. 9079 */ 9080 static int32_t 9081 ocs_hw_mq_process(ocs_hw_t *hw, int32_t status, sli4_queue_t *mq) 9082 { 9083 uint8_t mqe[SLI4_BMBX_SIZE]; 9084 9085 if (!sli_queue_read(&hw->sli, mq, mqe)) { 9086 ocs_hw_command_process(hw, status, mqe, mq->size); 9087 } 9088 9089 return 0; 9090 } 9091 9092 /** 9093 * @brief Read a FCF table entry. 9094 * 9095 * @param hw Hardware context. 9096 * @param index Table index to read. Use SLI4_FCOE_FCF_TABLE_FIRST for the first 9097 * read and the next_index field from the FCOE_READ_FCF_TABLE command 9098 * for subsequent reads. 9099 * 9100 * @return Returns 0 on success, or a non-zero value on failure. 9101 */ 9102 static ocs_hw_rtn_e 9103 ocs_hw_read_fcf(ocs_hw_t *hw, uint32_t index) 9104 { 9105 uint8_t *buf = NULL; 9106 int32_t rc = OCS_HW_RTN_ERROR; 9107 9108 buf = ocs_malloc(hw->os, SLI4_BMBX_SIZE, OCS_M_NOWAIT); 9109 if (!buf) { 9110 ocs_log_err(hw->os, "no buffer for command\n"); 9111 return OCS_HW_RTN_NO_MEMORY; 9112 } 9113 9114 if (sli_cmd_fcoe_read_fcf_table(&hw->sli, buf, SLI4_BMBX_SIZE, &hw->fcf_dmem, 9115 index)) { 9116 rc = ocs_hw_command(hw, buf, OCS_CMD_NOWAIT, ocs_hw_cb_read_fcf, &hw->fcf_dmem); 9117 } 9118 9119 if (rc != OCS_HW_RTN_SUCCESS) { 9120 ocs_log_test(hw->os, "FCOE_READ_FCF_TABLE failed\n"); 9121 ocs_free(hw->os, buf, SLI4_BMBX_SIZE); 9122 } 9123 9124 return rc; 9125 } 9126 9127 /** 9128 * @brief Callback function for the FCOE_READ_FCF_TABLE command. 9129 * 9130 * @par Description 9131 * Note that the caller has allocated: 9132 * - DMA memory to hold the table contents 9133 * - DMA memory structure 9134 * - Command/results buffer 9135 * . 9136 * Each of these must be freed here. 9137 * 9138 * @param hw Hardware context. 9139 * @param status Hardware status. 9140 * @param mqe Pointer to the mailbox command/results buffer. 9141 * @param arg Pointer to the DMA memory structure. 9142 * 9143 * @return Returns 0 on success, or a non-zero value on failure. 9144 */ 9145 static int32_t 9146 ocs_hw_cb_read_fcf(ocs_hw_t *hw, int32_t status, uint8_t *mqe, void *arg) 9147 { 9148 ocs_dma_t *dma = arg; 9149 sli4_mbox_command_header_t *hdr = (sli4_mbox_command_header_t *)mqe; 9150 9151 if (status || hdr->status) { 9152 ocs_log_test(hw->os, "bad status cqe=%#x mqe=%#x\n", 9153 status, hdr->status); 9154 } else if (dma->virt) { 9155 sli4_res_fcoe_read_fcf_table_t *read_fcf = dma->virt; 9156 9157 /* if FC or FCOE and FCF entry valid, process it */ 9158 if (read_fcf->fcf_entry.fc || 9159 (read_fcf->fcf_entry.val && !read_fcf->fcf_entry.sol)) { 9160 if (hw->callback.domain != NULL) { 9161 ocs_domain_record_t drec = {0}; 9162 9163 if (read_fcf->fcf_entry.fc) { 9164 /* 9165 * This is a pseudo FCF entry. Create a domain 9166 * record based on the read topology information 9167 */ 9168 drec.speed = hw->link.speed; 9169 drec.fc_id = hw->link.fc_id; 9170 drec.is_fc = TRUE; 9171 if (SLI_LINK_TOPO_LOOP == hw->link.topology) { 9172 drec.is_loop = TRUE; 9173 ocs_memcpy(drec.map.loop, hw->link.loop_map, 9174 sizeof(drec.map.loop)); 9175 } else if (SLI_LINK_TOPO_NPORT == hw->link.topology) { 9176 drec.is_nport = TRUE; 9177 } 9178 } else { 9179 drec.index = read_fcf->fcf_entry.fcf_index; 9180 drec.priority = read_fcf->fcf_entry.fip_priority; 9181 9182 /* copy address, wwn and vlan_bitmap */ 9183 ocs_memcpy(drec.address, read_fcf->fcf_entry.fcf_mac_address, 9184 sizeof(drec.address)); 9185 ocs_memcpy(drec.wwn, read_fcf->fcf_entry.fabric_name_id, 9186 sizeof(drec.wwn)); 9187 ocs_memcpy(drec.map.vlan, read_fcf->fcf_entry.vlan_bitmap, 9188 sizeof(drec.map.vlan)); 9189 9190 drec.is_ethernet = TRUE; 9191 drec.is_nport = TRUE; 9192 } 9193 9194 hw->callback.domain(hw->args.domain, 9195 OCS_HW_DOMAIN_FOUND, 9196 &drec); 9197 } 9198 } else { 9199 /* if FCOE and FCF is not valid, ignore it */ 9200 ocs_log_test(hw->os, "ignore invalid FCF entry\n"); 9201 } 9202 9203 if (SLI4_FCOE_FCF_TABLE_LAST != read_fcf->next_index) { 9204 ocs_hw_read_fcf(hw, read_fcf->next_index); 9205 } 9206 } 9207 9208 ocs_free(hw->os, mqe, SLI4_BMBX_SIZE); 9209 //ocs_dma_free(hw->os, dma); 9210 //ocs_free(hw->os, dma, sizeof(ocs_dma_t)); 9211 9212 return 0; 9213 } 9214 9215 /** 9216 * @brief Callback function for the SLI link events. 9217 * 9218 * @par Description 9219 * This function allocates memory which must be freed in its callback. 9220 * 9221 * @param ctx Hardware context pointer (that is, ocs_hw_t *). 9222 * @param e Event structure pointer (that is, sli4_link_event_t *). 9223 * 9224 * @return Returns 0 on success, or a non-zero value on failure. 9225 */ 9226 static int32_t 9227 ocs_hw_cb_link(void *ctx, void *e) 9228 { 9229 ocs_hw_t *hw = ctx; 9230 sli4_link_event_t *event = e; 9231 ocs_domain_t *d = NULL; 9232 uint32_t i = 0; 9233 int32_t rc = OCS_HW_RTN_ERROR; 9234 ocs_t *ocs = hw->os; 9235 9236 ocs_hw_link_event_init(hw); 9237 9238 switch (event->status) { 9239 case SLI_LINK_STATUS_UP: 9240 9241 hw->link = *event; 9242 9243 if (SLI_LINK_TOPO_NPORT == event->topology) { 9244 device_printf(ocs->dev, "Link Up, NPORT, speed is %d\n", event->speed); 9245 ocs_hw_read_fcf(hw, SLI4_FCOE_FCF_TABLE_FIRST); 9246 } else if (SLI_LINK_TOPO_LOOP == event->topology) { 9247 uint8_t *buf = NULL; 9248 device_printf(ocs->dev, "Link Up, LOOP, speed is %d\n", event->speed); 9249 9250 buf = ocs_malloc(hw->os, SLI4_BMBX_SIZE, OCS_M_NOWAIT); 9251 if (!buf) { 9252 ocs_log_err(hw->os, "no buffer for command\n"); 9253 break; 9254 } 9255 9256 if (sli_cmd_read_topology(&hw->sli, buf, SLI4_BMBX_SIZE, &hw->loop_map)) { 9257 rc = ocs_hw_command(hw, buf, OCS_CMD_NOWAIT, __ocs_read_topology_cb, NULL); 9258 } 9259 9260 if (rc != OCS_HW_RTN_SUCCESS) { 9261 ocs_log_test(hw->os, "READ_TOPOLOGY failed\n"); 9262 ocs_free(hw->os, buf, SLI4_BMBX_SIZE); 9263 } 9264 } else { 9265 device_printf(ocs->dev, "Link Up, unsupported topology (%#x), speed is %d\n", 9266 event->topology, event->speed); 9267 } 9268 break; 9269 case SLI_LINK_STATUS_DOWN: 9270 device_printf(ocs->dev, "Link Down\n"); 9271 9272 hw->link.status = event->status; 9273 9274 for (i = 0; i < SLI4_MAX_FCFI; i++) { 9275 d = hw->domains[i]; 9276 if (d != NULL && 9277 hw->callback.domain != NULL) { 9278 hw->callback.domain(hw->args.domain, OCS_HW_DOMAIN_LOST, d); 9279 } 9280 } 9281 break; 9282 default: 9283 ocs_log_test(hw->os, "unhandled link status %#x\n", event->status); 9284 break; 9285 } 9286 9287 return 0; 9288 } 9289 9290 static int32_t 9291 ocs_hw_cb_fip(void *ctx, void *e) 9292 { 9293 ocs_hw_t *hw = ctx; 9294 ocs_domain_t *domain = NULL; 9295 sli4_fip_event_t *event = e; 9296 9297 ocs_hw_assert(event); 9298 ocs_hw_assert(hw); 9299 9300 /* Find the associated domain object */ 9301 if (event->type == SLI4_FCOE_FIP_FCF_CLEAR_VLINK) { 9302 ocs_domain_t *d = NULL; 9303 uint32_t i = 0; 9304 9305 /* Clear VLINK is different from the other FIP events as it passes back 9306 * a VPI instead of a FCF index. Check all attached SLI ports for a 9307 * matching VPI */ 9308 for (i = 0; i < SLI4_MAX_FCFI; i++) { 9309 d = hw->domains[i]; 9310 if (d != NULL) { 9311 ocs_sport_t *sport = NULL; 9312 9313 ocs_list_foreach(&d->sport_list, sport) { 9314 if (sport->indicator == event->index) { 9315 domain = d; 9316 break; 9317 } 9318 } 9319 9320 if (domain != NULL) { 9321 break; 9322 } 9323 } 9324 } 9325 } else { 9326 domain = ocs_hw_domain_get_indexed(hw, event->index); 9327 } 9328 9329 switch (event->type) { 9330 case SLI4_FCOE_FIP_FCF_DISCOVERED: 9331 ocs_hw_read_fcf(hw, event->index); 9332 break; 9333 case SLI4_FCOE_FIP_FCF_DEAD: 9334 if (domain != NULL && 9335 hw->callback.domain != NULL) { 9336 hw->callback.domain(hw->args.domain, OCS_HW_DOMAIN_LOST, domain); 9337 } 9338 break; 9339 case SLI4_FCOE_FIP_FCF_CLEAR_VLINK: 9340 if (domain != NULL && 9341 hw->callback.domain != NULL) { 9342 /* 9343 * We will want to issue rediscover FCF when this domain is free'd in order 9344 * to invalidate the FCF table 9345 */ 9346 domain->req_rediscover_fcf = TRUE; 9347 hw->callback.domain(hw->args.domain, OCS_HW_DOMAIN_LOST, domain); 9348 } 9349 break; 9350 case SLI4_FCOE_FIP_FCF_MODIFIED: 9351 if (domain != NULL && 9352 hw->callback.domain != NULL) { 9353 hw->callback.domain(hw->args.domain, OCS_HW_DOMAIN_LOST, domain); 9354 } 9355 9356 ocs_hw_read_fcf(hw, event->index); 9357 break; 9358 default: 9359 ocs_log_test(hw->os, "unsupported event %#x\n", event->type); 9360 } 9361 9362 return 0; 9363 } 9364 9365 static int32_t 9366 ocs_hw_cb_node_attach(ocs_hw_t *hw, int32_t status, uint8_t *mqe, void *arg) 9367 { 9368 ocs_remote_node_t *rnode = arg; 9369 sli4_mbox_command_header_t *hdr = (sli4_mbox_command_header_t *)mqe; 9370 ocs_hw_remote_node_event_e evt = 0; 9371 9372 if (status || hdr->status) { 9373 ocs_log_debug(hw->os, "bad status cqe=%#x mqe=%#x\n", status, 9374 hdr->status); 9375 ocs_atomic_sub_return(&hw->rpi_ref[rnode->index].rpi_count, 1); 9376 rnode->attached = FALSE; 9377 ocs_atomic_set(&hw->rpi_ref[rnode->index].rpi_attached, 0); 9378 evt = OCS_HW_NODE_ATTACH_FAIL; 9379 } else { 9380 rnode->attached = TRUE; 9381 ocs_atomic_set(&hw->rpi_ref[rnode->index].rpi_attached, 1); 9382 evt = OCS_HW_NODE_ATTACH_OK; 9383 } 9384 9385 if (hw->callback.rnode != NULL) { 9386 hw->callback.rnode(hw->args.rnode, evt, rnode); 9387 } 9388 ocs_free(hw->os, mqe, SLI4_BMBX_SIZE); 9389 9390 return 0; 9391 } 9392 9393 static int32_t 9394 ocs_hw_cb_node_free(ocs_hw_t *hw, int32_t status, uint8_t *mqe, void *arg) 9395 { 9396 ocs_remote_node_t *rnode = arg; 9397 sli4_mbox_command_header_t *hdr = (sli4_mbox_command_header_t *)mqe; 9398 ocs_hw_remote_node_event_e evt = OCS_HW_NODE_FREE_FAIL; 9399 int32_t rc = 0; 9400 9401 if (status || hdr->status) { 9402 ocs_log_debug(hw->os, "bad status cqe=%#x mqe=%#x\n", status, 9403 hdr->status); 9404 9405 /* 9406 * In certain cases, a non-zero MQE status is OK (all must be true): 9407 * - node is attached 9408 * - if High Login Mode is enabled, node is part of a node group 9409 * - status is 0x1400 9410 */ 9411 if (!rnode->attached || ((sli_get_hlm(&hw->sli) == TRUE) && !rnode->node_group) || 9412 (hdr->status != SLI4_MBOX_STATUS_RPI_NOT_REG)) { 9413 rc = -1; 9414 } 9415 } 9416 9417 if (rc == 0) { 9418 rnode->node_group = FALSE; 9419 rnode->attached = FALSE; 9420 9421 if (ocs_atomic_read(&hw->rpi_ref[rnode->index].rpi_count) == 0) { 9422 ocs_atomic_set(&hw->rpi_ref[rnode->index].rpi_attached, 0); 9423 } 9424 9425 evt = OCS_HW_NODE_FREE_OK; 9426 } 9427 9428 if (hw->callback.rnode != NULL) { 9429 hw->callback.rnode(hw->args.rnode, evt, rnode); 9430 } 9431 9432 ocs_free(hw->os, mqe, SLI4_BMBX_SIZE); 9433 9434 return rc; 9435 } 9436 9437 static int32_t 9438 ocs_hw_cb_node_free_all(ocs_hw_t *hw, int32_t status, uint8_t *mqe, void *arg) 9439 { 9440 sli4_mbox_command_header_t *hdr = (sli4_mbox_command_header_t *)mqe; 9441 ocs_hw_remote_node_event_e evt = OCS_HW_NODE_FREE_FAIL; 9442 int32_t rc = 0; 9443 uint32_t i; 9444 9445 if (status || hdr->status) { 9446 ocs_log_debug(hw->os, "bad status cqe=%#x mqe=%#x\n", status, 9447 hdr->status); 9448 } else { 9449 evt = OCS_HW_NODE_FREE_ALL_OK; 9450 } 9451 9452 if (evt == OCS_HW_NODE_FREE_ALL_OK) { 9453 for (i = 0; i < sli_get_max_rsrc(&hw->sli, SLI_RSRC_FCOE_RPI); i++) { 9454 ocs_atomic_set(&hw->rpi_ref[i].rpi_count, 0); 9455 } 9456 9457 if (sli_resource_reset(&hw->sli, SLI_RSRC_FCOE_RPI)) { 9458 ocs_log_test(hw->os, "FCOE_RPI free all failure\n"); 9459 rc = -1; 9460 } 9461 } 9462 9463 if (hw->callback.rnode != NULL) { 9464 hw->callback.rnode(hw->args.rnode, evt, NULL); 9465 } 9466 9467 ocs_free(hw->os, mqe, SLI4_BMBX_SIZE); 9468 9469 return rc; 9470 } 9471 9472 /** 9473 * @brief Initialize the pool of HW IO objects. 9474 * 9475 * @param hw Hardware context. 9476 * 9477 * @return Returns 0 on success, or a non-zero value on failure. 9478 */ 9479 static ocs_hw_rtn_e 9480 ocs_hw_setup_io(ocs_hw_t *hw) 9481 { 9482 uint32_t i = 0; 9483 ocs_hw_io_t *io = NULL; 9484 uintptr_t xfer_virt = 0; 9485 uintptr_t xfer_phys = 0; 9486 uint32_t index; 9487 uint8_t new_alloc = TRUE; 9488 9489 if (NULL == hw->io) { 9490 hw->io = ocs_malloc(hw->os, hw->config.n_io * sizeof(ocs_hw_io_t *), OCS_M_ZERO | OCS_M_NOWAIT); 9491 9492 if (NULL == hw->io) { 9493 ocs_log_err(hw->os, "IO pointer memory allocation failed, %d Ios at size %zu\n", 9494 hw->config.n_io, 9495 sizeof(ocs_hw_io_t *)); 9496 return OCS_HW_RTN_NO_MEMORY; 9497 } 9498 for (i = 0; i < hw->config.n_io; i++) { 9499 hw->io[i] = ocs_malloc(hw->os, sizeof(ocs_hw_io_t), 9500 OCS_M_ZERO | OCS_M_NOWAIT); 9501 if (hw->io[i] == NULL) { 9502 ocs_log_err(hw->os, "IO(%d) memory allocation failed\n", i); 9503 goto error; 9504 } 9505 } 9506 9507 /* Create WQE buffs for IO */ 9508 hw->wqe_buffs = ocs_malloc(hw->os, hw->config.n_io * hw->sli.config.wqe_size, 9509 OCS_M_ZERO | OCS_M_NOWAIT); 9510 if (NULL == hw->wqe_buffs) { 9511 ocs_free(hw->os, hw->io, hw->config.n_io * sizeof(ocs_hw_io_t)); 9512 ocs_log_err(hw->os, "%s: IO WQE buff allocation failed, %d Ios at size %zu\n", 9513 __func__, hw->config.n_io, hw->sli.config.wqe_size); 9514 return OCS_HW_RTN_NO_MEMORY; 9515 } 9516 9517 } else { 9518 /* re-use existing IOs, including SGLs */ 9519 new_alloc = FALSE; 9520 } 9521 9522 if (new_alloc) { 9523 if (ocs_dma_alloc(hw->os, &hw->xfer_rdy, 9524 sizeof(fcp_xfer_rdy_iu_t) * hw->config.n_io, 9525 4/*XXX what does this need to be? */)) { 9526 ocs_log_err(hw->os, "XFER_RDY buffer allocation failed\n"); 9527 return OCS_HW_RTN_NO_MEMORY; 9528 } 9529 } 9530 xfer_virt = (uintptr_t)hw->xfer_rdy.virt; 9531 xfer_phys = hw->xfer_rdy.phys; 9532 9533 for (i = 0; i < hw->config.n_io; i++) { 9534 hw_wq_callback_t *wqcb; 9535 9536 io = hw->io[i]; 9537 9538 /* initialize IO fields */ 9539 io->hw = hw; 9540 9541 /* Assign a WQE buff */ 9542 io->wqe.wqebuf = &hw->wqe_buffs[i * hw->sli.config.wqe_size]; 9543 9544 /* Allocate the request tag for this IO */ 9545 wqcb = ocs_hw_reqtag_alloc(hw, ocs_hw_wq_process_io, io); 9546 if (wqcb == NULL) { 9547 ocs_log_err(hw->os, "can't allocate request tag\n"); 9548 return OCS_HW_RTN_NO_RESOURCES; 9549 } 9550 io->reqtag = wqcb->instance_index; 9551 9552 /* Now for the fields that are initialized on each free */ 9553 ocs_hw_init_free_io(io); 9554 9555 /* The XB flag isn't cleared on IO free, so initialize it to zero here */ 9556 io->xbusy = 0; 9557 9558 if (sli_resource_alloc(&hw->sli, SLI_RSRC_FCOE_XRI, &io->indicator, &index)) { 9559 ocs_log_err(hw->os, "sli_resource_alloc failed @ %d\n", i); 9560 return OCS_HW_RTN_NO_MEMORY; 9561 } 9562 9563 if (new_alloc && ocs_dma_alloc(hw->os, &io->def_sgl, hw->config.n_sgl * sizeof(sli4_sge_t), 64)) { 9564 ocs_log_err(hw->os, "ocs_dma_alloc failed @ %d\n", i); 9565 ocs_memset(&io->def_sgl, 0, sizeof(ocs_dma_t)); 9566 return OCS_HW_RTN_NO_MEMORY; 9567 } 9568 io->def_sgl_count = hw->config.n_sgl; 9569 io->sgl = &io->def_sgl; 9570 io->sgl_count = io->def_sgl_count; 9571 9572 if (hw->xfer_rdy.size) { 9573 io->xfer_rdy.virt = (void *)xfer_virt; 9574 io->xfer_rdy.phys = xfer_phys; 9575 io->xfer_rdy.size = sizeof(fcp_xfer_rdy_iu_t); 9576 9577 xfer_virt += sizeof(fcp_xfer_rdy_iu_t); 9578 xfer_phys += sizeof(fcp_xfer_rdy_iu_t); 9579 } 9580 } 9581 9582 return OCS_HW_RTN_SUCCESS; 9583 error: 9584 for (i = 0; i < hw->config.n_io && hw->io[i]; i++) { 9585 ocs_free(hw->os, hw->io[i], sizeof(ocs_hw_io_t)); 9586 hw->io[i] = NULL; 9587 } 9588 9589 return OCS_HW_RTN_NO_MEMORY; 9590 } 9591 9592 static ocs_hw_rtn_e 9593 ocs_hw_init_io(ocs_hw_t *hw) 9594 { 9595 uint32_t i = 0, io_index = 0; 9596 uint32_t prereg = 0; 9597 ocs_hw_io_t *io = NULL; 9598 uint8_t cmd[SLI4_BMBX_SIZE]; 9599 ocs_hw_rtn_e rc = OCS_HW_RTN_SUCCESS; 9600 uint32_t nremaining; 9601 uint32_t n = 0; 9602 uint32_t sgls_per_request = 256; 9603 ocs_dma_t **sgls = NULL; 9604 ocs_dma_t reqbuf = { 0 }; 9605 9606 prereg = sli_get_sgl_preregister(&hw->sli); 9607 9608 if (prereg) { 9609 sgls = ocs_malloc(hw->os, sizeof(*sgls) * sgls_per_request, OCS_M_NOWAIT); 9610 if (sgls == NULL) { 9611 ocs_log_err(hw->os, "ocs_malloc sgls failed\n"); 9612 return OCS_HW_RTN_NO_MEMORY; 9613 } 9614 9615 rc = ocs_dma_alloc(hw->os, &reqbuf, 32 + sgls_per_request*16, OCS_MIN_DMA_ALIGNMENT); 9616 if (rc) { 9617 ocs_log_err(hw->os, "ocs_dma_alloc reqbuf failed\n"); 9618 ocs_free(hw->os, sgls, sizeof(*sgls) * sgls_per_request); 9619 return OCS_HW_RTN_NO_MEMORY; 9620 } 9621 } 9622 9623 io = hw->io[io_index]; 9624 for (nremaining = hw->config.n_io; nremaining; nremaining -= n) { 9625 if (prereg) { 9626 /* Copy address of SGL's into local sgls[] array, break out if the xri 9627 * is not contiguous. 9628 */ 9629 for (n = 0; n < MIN(sgls_per_request, nremaining); n++) { 9630 /* Check that we have contiguous xri values */ 9631 if (n > 0) { 9632 if (hw->io[io_index + n]->indicator != (hw->io[io_index + n-1]->indicator+1)) { 9633 break; 9634 } 9635 } 9636 sgls[n] = hw->io[io_index + n]->sgl; 9637 } 9638 9639 if (sli_cmd_fcoe_post_sgl_pages(&hw->sli, cmd, sizeof(cmd), 9640 io->indicator, n, sgls, NULL, &reqbuf)) { 9641 if (ocs_hw_command(hw, cmd, OCS_CMD_POLL, NULL, NULL)) { 9642 rc = OCS_HW_RTN_ERROR; 9643 ocs_log_err(hw->os, "SGL post failed\n"); 9644 break; 9645 } 9646 } 9647 } else { 9648 n = nremaining; 9649 } 9650 9651 /* Add to tail if successful */ 9652 for (i = 0; i < n; i ++) { 9653 io->is_port_owned = 0; 9654 io->state = OCS_HW_IO_STATE_FREE; 9655 ocs_list_add_tail(&hw->io_free, io); 9656 io = hw->io[io_index+1]; 9657 io_index++; 9658 } 9659 } 9660 9661 if (prereg) { 9662 ocs_dma_free(hw->os, &reqbuf); 9663 ocs_free(hw->os, sgls, sizeof(*sgls) * sgls_per_request); 9664 } 9665 9666 return rc; 9667 } 9668 9669 static int32_t 9670 ocs_hw_flush(ocs_hw_t *hw) 9671 { 9672 uint32_t i = 0; 9673 9674 /* Process any remaining completions */ 9675 for (i = 0; i < hw->eq_count; i++) { 9676 ocs_hw_process(hw, i, ~0); 9677 } 9678 9679 return 0; 9680 } 9681 9682 static int32_t 9683 ocs_hw_command_cancel(ocs_hw_t *hw) 9684 { 9685 9686 ocs_lock(&hw->cmd_lock); 9687 9688 /* 9689 * Manually clean up remaining commands. Note: since this calls 9690 * ocs_hw_command_process(), we'll also process the cmd_pending 9691 * list, so no need to manually clean that out. 9692 */ 9693 while (!ocs_list_empty(&hw->cmd_head)) { 9694 uint8_t mqe[SLI4_BMBX_SIZE] = { 0 }; 9695 ocs_command_ctx_t *ctx = ocs_list_get_head(&hw->cmd_head); 9696 9697 ocs_log_test(hw->os, "hung command %08x\n", 9698 NULL == ctx ? UINT32_MAX : 9699 (NULL == ctx->buf ? UINT32_MAX : *((uint32_t *)ctx->buf))); 9700 ocs_unlock(&hw->cmd_lock); 9701 ocs_hw_command_process(hw, -1/*Bad status*/, mqe, SLI4_BMBX_SIZE); 9702 ocs_lock(&hw->cmd_lock); 9703 } 9704 9705 ocs_unlock(&hw->cmd_lock); 9706 9707 return 0; 9708 } 9709 9710 /** 9711 * @brief Find IO given indicator (xri). 9712 * 9713 * @param hw Hal context. 9714 * @param indicator Indicator (xri) to look for. 9715 * 9716 * @return Returns io if found, NULL otherwise. 9717 */ 9718 ocs_hw_io_t * 9719 ocs_hw_io_lookup(ocs_hw_t *hw, uint32_t xri) 9720 { 9721 uint32_t ioindex; 9722 ioindex = xri - hw->sli.config.extent[SLI_RSRC_FCOE_XRI].base[0]; 9723 return hw->io[ioindex]; 9724 } 9725 9726 /** 9727 * @brief Issue any pending callbacks for an IO and remove off the timer and pending lists. 9728 * 9729 * @param hw Hal context. 9730 * @param io Pointer to the IO to cleanup. 9731 */ 9732 static void 9733 ocs_hw_io_cancel_cleanup(ocs_hw_t *hw, ocs_hw_io_t *io) 9734 { 9735 ocs_hw_done_t done = io->done; 9736 ocs_hw_done_t abort_done = io->abort_done; 9737 9738 /* first check active_wqe list and remove if there */ 9739 if (ocs_list_on_list(&io->wqe_link)) { 9740 ocs_list_remove(&hw->io_timed_wqe, io); 9741 } 9742 9743 /* Remove from WQ pending list */ 9744 if ((io->wq != NULL) && ocs_list_on_list(&io->wq->pending_list)) { 9745 ocs_list_remove(&io->wq->pending_list, io); 9746 } 9747 9748 if (io->done) { 9749 void *arg = io->arg; 9750 9751 io->done = NULL; 9752 ocs_unlock(&hw->io_lock); 9753 done(io, io->rnode, 0, SLI4_FC_WCQE_STATUS_SHUTDOWN, 0, arg); 9754 ocs_lock(&hw->io_lock); 9755 } 9756 9757 if (io->abort_done != NULL) { 9758 void *abort_arg = io->abort_arg; 9759 9760 io->abort_done = NULL; 9761 ocs_unlock(&hw->io_lock); 9762 abort_done(io, io->rnode, 0, SLI4_FC_WCQE_STATUS_SHUTDOWN, 0, abort_arg); 9763 ocs_lock(&hw->io_lock); 9764 } 9765 } 9766 9767 static int32_t 9768 ocs_hw_io_cancel(ocs_hw_t *hw) 9769 { 9770 ocs_hw_io_t *io = NULL; 9771 ocs_hw_io_t *tmp_io = NULL; 9772 uint32_t iters = 100; /* One second limit */ 9773 9774 /* 9775 * Manually clean up outstanding IO. 9776 * Only walk through list once: the backend will cleanup any IOs when done/abort_done is called. 9777 */ 9778 ocs_lock(&hw->io_lock); 9779 ocs_list_foreach_safe(&hw->io_inuse, io, tmp_io) { 9780 ocs_hw_done_t done = io->done; 9781 ocs_hw_done_t abort_done = io->abort_done; 9782 9783 ocs_hw_io_cancel_cleanup(hw, io); 9784 9785 /* 9786 * Since this is called in a reset/shutdown 9787 * case, If there is no callback, then just 9788 * free the IO. 9789 * 9790 * Note: A port owned XRI cannot be on 9791 * the in use list. We cannot call 9792 * ocs_hw_io_free() because we already 9793 * hold the io_lock. 9794 */ 9795 if (done == NULL && 9796 abort_done == NULL) { 9797 /* 9798 * Since this is called in a reset/shutdown 9799 * case, If there is no callback, then just 9800 * free the IO. 9801 */ 9802 ocs_hw_io_free_common(hw, io); 9803 ocs_list_remove(&hw->io_inuse, io); 9804 ocs_hw_io_free_move_correct_list(hw, io); 9805 } 9806 } 9807 9808 /* 9809 * For port owned XRIs, they are not on the in use list, so 9810 * walk though XRIs and issue any callbacks. 9811 */ 9812 ocs_list_foreach_safe(&hw->io_port_owned, io, tmp_io) { 9813 /* check list and remove if there */ 9814 if (ocs_list_on_list(&io->dnrx_link)) { 9815 ocs_list_remove(&hw->io_port_dnrx, io); 9816 ocs_ref_put(&io->ref); /* ocs_ref_get(): same function */ 9817 } 9818 ocs_hw_io_cancel_cleanup(hw, io); 9819 ocs_list_remove(&hw->io_port_owned, io); 9820 ocs_hw_io_free_common(hw, io); 9821 } 9822 ocs_unlock(&hw->io_lock); 9823 9824 /* Give time for the callbacks to complete */ 9825 do { 9826 ocs_udelay(10000); 9827 iters--; 9828 } while (!ocs_list_empty(&hw->io_inuse) && iters); 9829 9830 /* Leave a breadcrumb that cleanup is not yet complete. */ 9831 if (!ocs_list_empty(&hw->io_inuse)) { 9832 ocs_log_test(hw->os, "io_inuse list is not empty\n"); 9833 } 9834 9835 return 0; 9836 } 9837 9838 static int32_t 9839 ocs_hw_io_ini_sge(ocs_hw_t *hw, ocs_hw_io_t *io, ocs_dma_t *cmnd, uint32_t cmnd_size, 9840 ocs_dma_t *rsp) 9841 { 9842 sli4_sge_t *data = NULL; 9843 9844 if (!hw || !io) { 9845 ocs_log_err(NULL, "bad parm hw=%p io=%p\n", hw, io); 9846 return OCS_HW_RTN_ERROR; 9847 } 9848 9849 data = io->def_sgl.virt; 9850 9851 /* setup command pointer */ 9852 data->buffer_address_high = ocs_addr32_hi(cmnd->phys); 9853 data->buffer_address_low = ocs_addr32_lo(cmnd->phys); 9854 data->buffer_length = cmnd_size; 9855 data++; 9856 9857 /* setup response pointer */ 9858 data->buffer_address_high = ocs_addr32_hi(rsp->phys); 9859 data->buffer_address_low = ocs_addr32_lo(rsp->phys); 9860 data->buffer_length = rsp->size; 9861 9862 return 0; 9863 } 9864 9865 static int32_t 9866 __ocs_read_topology_cb(ocs_hw_t *hw, int32_t status, uint8_t *mqe, void *arg) 9867 { 9868 sli4_cmd_read_topology_t *read_topo = (sli4_cmd_read_topology_t *)mqe; 9869 9870 if (status || read_topo->hdr.status) { 9871 ocs_log_debug(hw->os, "bad status cqe=%#x mqe=%#x\n", 9872 status, read_topo->hdr.status); 9873 ocs_free(hw->os, mqe, SLI4_BMBX_SIZE); 9874 return -1; 9875 } 9876 9877 switch (read_topo->attention_type) { 9878 case SLI4_READ_TOPOLOGY_LINK_UP: 9879 hw->link.status = SLI_LINK_STATUS_UP; 9880 break; 9881 case SLI4_READ_TOPOLOGY_LINK_DOWN: 9882 hw->link.status = SLI_LINK_STATUS_DOWN; 9883 break; 9884 case SLI4_READ_TOPOLOGY_LINK_NO_ALPA: 9885 hw->link.status = SLI_LINK_STATUS_NO_ALPA; 9886 break; 9887 default: 9888 hw->link.status = SLI_LINK_STATUS_MAX; 9889 break; 9890 } 9891 9892 switch (read_topo->topology) { 9893 case SLI4_READ_TOPOLOGY_NPORT: 9894 hw->link.topology = SLI_LINK_TOPO_NPORT; 9895 break; 9896 case SLI4_READ_TOPOLOGY_FC_AL: 9897 hw->link.topology = SLI_LINK_TOPO_LOOP; 9898 if (SLI_LINK_STATUS_UP == hw->link.status) { 9899 hw->link.loop_map = hw->loop_map.virt; 9900 } 9901 hw->link.fc_id = read_topo->acquired_al_pa; 9902 break; 9903 default: 9904 hw->link.topology = SLI_LINK_TOPO_MAX; 9905 break; 9906 } 9907 9908 hw->link.medium = SLI_LINK_MEDIUM_FC; 9909 9910 switch (read_topo->link_current.link_speed) { 9911 case SLI4_READ_TOPOLOGY_SPEED_1G: 9912 hw->link.speed = 1 * 1000; 9913 break; 9914 case SLI4_READ_TOPOLOGY_SPEED_2G: 9915 hw->link.speed = 2 * 1000; 9916 break; 9917 case SLI4_READ_TOPOLOGY_SPEED_4G: 9918 hw->link.speed = 4 * 1000; 9919 break; 9920 case SLI4_READ_TOPOLOGY_SPEED_8G: 9921 hw->link.speed = 8 * 1000; 9922 break; 9923 case SLI4_READ_TOPOLOGY_SPEED_16G: 9924 hw->link.speed = 16 * 1000; 9925 hw->link.loop_map = NULL; 9926 break; 9927 case SLI4_READ_TOPOLOGY_SPEED_32G: 9928 hw->link.speed = 32 * 1000; 9929 hw->link.loop_map = NULL; 9930 break; 9931 } 9932 9933 ocs_free(hw->os, mqe, SLI4_BMBX_SIZE); 9934 9935 ocs_hw_read_fcf(hw, SLI4_FCOE_FCF_TABLE_FIRST); 9936 9937 return 0; 9938 } 9939 9940 static int32_t 9941 __ocs_hw_port_common(const char *funcname, ocs_sm_ctx_t *ctx, ocs_sm_event_t evt, void *data) 9942 { 9943 ocs_sli_port_t *sport = ctx->app; 9944 ocs_hw_t *hw = sport->hw; 9945 9946 smtrace("port"); 9947 9948 switch (evt) { 9949 case OCS_EVT_EXIT: 9950 /* ignore */ 9951 break; 9952 9953 case OCS_EVT_HW_PORT_REQ_FREE: 9954 case OCS_EVT_HW_PORT_REQ_ATTACH: 9955 if (data != NULL) { 9956 ocs_free(hw->os, data, SLI4_BMBX_SIZE); 9957 } 9958 /* fall through */ 9959 default: 9960 ocs_log_test(hw->os, "%s %-20s not handled\n", funcname, ocs_sm_event_name(evt)); 9961 break; 9962 } 9963 9964 return 0; 9965 } 9966 9967 static void * 9968 __ocs_hw_port_free_report_fail(ocs_sm_ctx_t *ctx, ocs_sm_event_t evt, void *data) 9969 { 9970 ocs_sli_port_t *sport = ctx->app; 9971 ocs_hw_t *hw = sport->hw; 9972 9973 smtrace("port"); 9974 9975 switch (evt) { 9976 case OCS_EVT_ENTER: 9977 if (data != NULL) { 9978 ocs_free(hw->os, data, SLI4_BMBX_SIZE); 9979 } 9980 if (hw->callback.port != NULL) { 9981 hw->callback.port(hw->args.port, 9982 OCS_HW_PORT_FREE_FAIL, sport); 9983 } 9984 break; 9985 default: 9986 break; 9987 } 9988 9989 return NULL; 9990 } 9991 9992 static void * 9993 __ocs_hw_port_freed(ocs_sm_ctx_t *ctx, ocs_sm_event_t evt, void *data) 9994 { 9995 ocs_sli_port_t *sport = ctx->app; 9996 ocs_hw_t *hw = sport->hw; 9997 9998 smtrace("port"); 9999 10000 switch (evt) { 10001 case OCS_EVT_ENTER: 10002 /* free SLI resource */ 10003 if (sli_resource_free(&hw->sli, SLI_RSRC_FCOE_VPI, sport->indicator)) { 10004 ocs_log_err(hw->os, "FCOE_VPI free failure addr=%#x\n", sport->fc_id); 10005 } 10006 10007 /* free mailbox buffer */ 10008 if (data != NULL) { 10009 ocs_free(hw->os, data, SLI4_BMBX_SIZE); 10010 } 10011 if (hw->callback.port != NULL) { 10012 hw->callback.port(hw->args.port, 10013 OCS_HW_PORT_FREE_OK, sport); 10014 } 10015 break; 10016 default: 10017 break; 10018 } 10019 10020 return NULL; 10021 } 10022 10023 static void * 10024 __ocs_hw_port_attach_report_fail(ocs_sm_ctx_t *ctx, ocs_sm_event_t evt, void *data) 10025 { 10026 ocs_sli_port_t *sport = ctx->app; 10027 ocs_hw_t *hw = sport->hw; 10028 10029 smtrace("port"); 10030 10031 switch (evt) { 10032 case OCS_EVT_ENTER: 10033 /* free SLI resource */ 10034 sli_resource_free(&hw->sli, SLI_RSRC_FCOE_VPI, sport->indicator); 10035 10036 /* free mailbox buffer */ 10037 if (data != NULL) { 10038 ocs_free(hw->os, data, SLI4_BMBX_SIZE); 10039 } 10040 10041 if (hw->callback.port != NULL) { 10042 hw->callback.port(hw->args.port, 10043 OCS_HW_PORT_ATTACH_FAIL, sport); 10044 } 10045 if (sport->sm_free_req_pending) { 10046 ocs_sm_transition(ctx, __ocs_hw_port_free_unreg_vpi, NULL); 10047 } 10048 break; 10049 default: 10050 __ocs_hw_port_common(__func__, ctx, evt, data); 10051 break; 10052 } 10053 10054 return NULL; 10055 } 10056 10057 static void * 10058 __ocs_hw_port_free_unreg_vpi(ocs_sm_ctx_t *ctx, ocs_sm_event_t evt, void *data) 10059 { 10060 ocs_sli_port_t *sport = ctx->app; 10061 ocs_hw_t *hw = sport->hw; 10062 uint8_t *cmd = NULL; 10063 10064 smtrace("port"); 10065 10066 switch (evt) { 10067 case OCS_EVT_ENTER: 10068 /* allocate memory and send unreg_vpi */ 10069 cmd = ocs_malloc(hw->os, SLI4_BMBX_SIZE, OCS_M_ZERO | OCS_M_NOWAIT); 10070 if (!cmd) { 10071 ocs_sm_post_event(ctx, OCS_EVT_ERROR, NULL); 10072 break; 10073 } 10074 10075 if (0 == sli_cmd_unreg_vpi(&hw->sli, cmd, SLI4_BMBX_SIZE, sport->indicator, 10076 SLI4_UNREG_TYPE_PORT)) { 10077 ocs_log_err(hw->os, "UNREG_VPI format failure\n"); 10078 ocs_free(hw->os, cmd, SLI4_BMBX_SIZE); 10079 ocs_sm_post_event(ctx, OCS_EVT_ERROR, NULL); 10080 break; 10081 } 10082 10083 if (ocs_hw_command(hw, cmd, OCS_CMD_NOWAIT, __ocs_hw_port_cb, sport)) { 10084 ocs_log_err(hw->os, "UNREG_VPI command failure\n"); 10085 ocs_free(hw->os, cmd, SLI4_BMBX_SIZE); 10086 ocs_sm_post_event(ctx, OCS_EVT_ERROR, NULL); 10087 break; 10088 } 10089 break; 10090 case OCS_EVT_RESPONSE: 10091 ocs_sm_transition(ctx, __ocs_hw_port_freed, data); 10092 break; 10093 case OCS_EVT_ERROR: 10094 ocs_sm_transition(ctx, __ocs_hw_port_free_report_fail, data); 10095 break; 10096 default: 10097 __ocs_hw_port_common(__func__, ctx, evt, data); 10098 break; 10099 } 10100 10101 return NULL; 10102 } 10103 10104 static void * 10105 __ocs_hw_port_free_nop(ocs_sm_ctx_t *ctx, ocs_sm_event_t evt, void *data) 10106 { 10107 ocs_sli_port_t *sport = ctx->app; 10108 ocs_hw_t *hw = sport->hw; 10109 10110 smtrace("port"); 10111 10112 switch (evt) { 10113 case OCS_EVT_ENTER: 10114 /* Forward to execute in mailbox completion processing context */ 10115 if (ocs_hw_async_call(hw, __ocs_hw_port_realloc_cb, sport)) { 10116 ocs_log_err(hw->os, "ocs_hw_async_call failed\n"); 10117 } 10118 break; 10119 case OCS_EVT_RESPONSE: 10120 ocs_sm_transition(ctx, __ocs_hw_port_freed, data); 10121 break; 10122 case OCS_EVT_ERROR: 10123 ocs_sm_transition(ctx, __ocs_hw_port_free_report_fail, data); 10124 break; 10125 default: 10126 break; 10127 } 10128 10129 return NULL; 10130 } 10131 10132 static void * 10133 __ocs_hw_port_attached(ocs_sm_ctx_t *ctx, ocs_sm_event_t evt, void *data) 10134 { 10135 ocs_sli_port_t *sport = ctx->app; 10136 ocs_hw_t *hw = sport->hw; 10137 10138 smtrace("port"); 10139 10140 switch (evt) { 10141 case OCS_EVT_ENTER: 10142 if (data != NULL) { 10143 ocs_free(hw->os, data, SLI4_BMBX_SIZE); 10144 } 10145 if (hw->callback.port != NULL) { 10146 hw->callback.port(hw->args.port, 10147 OCS_HW_PORT_ATTACH_OK, sport); 10148 } 10149 if (sport->sm_free_req_pending) { 10150 ocs_sm_transition(ctx, __ocs_hw_port_free_unreg_vpi, NULL); 10151 } 10152 break; 10153 case OCS_EVT_HW_PORT_REQ_FREE: 10154 /* virtual/physical port request free */ 10155 ocs_sm_transition(ctx, __ocs_hw_port_free_unreg_vpi, NULL); 10156 break; 10157 default: 10158 __ocs_hw_port_common(__func__, ctx, evt, data); 10159 break; 10160 } 10161 10162 return NULL; 10163 } 10164 10165 static void * 10166 __ocs_hw_port_attach_reg_vpi(ocs_sm_ctx_t *ctx, ocs_sm_event_t evt, void *data) 10167 { 10168 ocs_sli_port_t *sport = ctx->app; 10169 ocs_hw_t *hw = sport->hw; 10170 10171 smtrace("port"); 10172 10173 switch (evt) { 10174 case OCS_EVT_ENTER: 10175 if (0 == sli_cmd_reg_vpi(&hw->sli, data, SLI4_BMBX_SIZE, sport, FALSE)) { 10176 ocs_log_err(hw->os, "REG_VPI format failure\n"); 10177 ocs_sm_post_event(ctx, OCS_EVT_ERROR, NULL); 10178 break; 10179 } 10180 10181 if (ocs_hw_command(hw, data, OCS_CMD_NOWAIT, __ocs_hw_port_cb, sport)) { 10182 ocs_log_err(hw->os, "REG_VPI command failure\n"); 10183 ocs_sm_post_event(ctx, OCS_EVT_ERROR, NULL); 10184 break; 10185 } 10186 break; 10187 case OCS_EVT_RESPONSE: 10188 ocs_sm_transition(ctx, __ocs_hw_port_attached, data); 10189 break; 10190 case OCS_EVT_ERROR: 10191 ocs_sm_transition(ctx, __ocs_hw_port_attach_report_fail, data); 10192 break; 10193 case OCS_EVT_HW_PORT_REQ_FREE: 10194 /* Wait for attach response and then free */ 10195 sport->sm_free_req_pending = 1; 10196 break; 10197 default: 10198 __ocs_hw_port_common(__func__, ctx, evt, data); 10199 break; 10200 } 10201 10202 return NULL; 10203 } 10204 10205 static void * 10206 __ocs_hw_port_done(ocs_sm_ctx_t *ctx, ocs_sm_event_t evt, void *data) 10207 { 10208 ocs_sli_port_t *sport = ctx->app; 10209 ocs_hw_t *hw = sport->hw; 10210 10211 smtrace("port"); 10212 10213 switch (evt) { 10214 case OCS_EVT_ENTER: 10215 /* free SLI resource */ 10216 sli_resource_free(&hw->sli, SLI_RSRC_FCOE_VPI, sport->indicator); 10217 10218 /* free mailbox buffer */ 10219 if (data != NULL) { 10220 ocs_free(hw->os, data, SLI4_BMBX_SIZE); 10221 } 10222 break; 10223 default: 10224 __ocs_hw_port_common(__func__, ctx, evt, data); 10225 break; 10226 } 10227 10228 return NULL; 10229 } 10230 10231 static void * 10232 __ocs_hw_port_allocated(ocs_sm_ctx_t *ctx, ocs_sm_event_t evt, void *data) 10233 { 10234 ocs_sli_port_t *sport = ctx->app; 10235 ocs_hw_t *hw = sport->hw; 10236 10237 smtrace("port"); 10238 10239 switch (evt) { 10240 case OCS_EVT_ENTER: 10241 if (data != NULL) { 10242 ocs_free(hw->os, data, SLI4_BMBX_SIZE); 10243 } 10244 if (hw->callback.port != NULL) { 10245 hw->callback.port(hw->args.port, 10246 OCS_HW_PORT_ALLOC_OK, sport); 10247 } 10248 /* If there is a pending free request, then handle it now */ 10249 if (sport->sm_free_req_pending) { 10250 ocs_sm_transition(ctx, __ocs_hw_port_free_unreg_vpi, NULL); 10251 } 10252 break; 10253 case OCS_EVT_HW_PORT_REQ_ATTACH: 10254 /* virtual port requests attach */ 10255 ocs_sm_transition(ctx, __ocs_hw_port_attach_reg_vpi, data); 10256 break; 10257 case OCS_EVT_HW_PORT_ATTACH_OK: 10258 /* physical port attached (as part of attaching domain) */ 10259 ocs_sm_transition(ctx, __ocs_hw_port_attached, data); 10260 break; 10261 case OCS_EVT_HW_PORT_REQ_FREE: 10262 /* virtual port request free */ 10263 if (SLI4_IF_TYPE_LANCER_FC_ETH == sli_get_if_type(&hw->sli)) { 10264 ocs_sm_transition(ctx, __ocs_hw_port_free_unreg_vpi, NULL); 10265 } else { 10266 /* 10267 * Note: BE3/Skyhawk will respond with a status of 0x20 10268 * unless the reg_vpi has been issued, so we can 10269 * skip the unreg_vpi for these adapters. 10270 * 10271 * Send a nop to make sure that free doesn't occur in 10272 * same context 10273 */ 10274 ocs_sm_transition(ctx, __ocs_hw_port_free_nop, NULL); 10275 } 10276 break; 10277 default: 10278 __ocs_hw_port_common(__func__, ctx, evt, data); 10279 break; 10280 } 10281 10282 return NULL; 10283 } 10284 10285 static void * 10286 __ocs_hw_port_alloc_report_fail(ocs_sm_ctx_t *ctx, ocs_sm_event_t evt, void *data) 10287 { 10288 ocs_sli_port_t *sport = ctx->app; 10289 ocs_hw_t *hw = sport->hw; 10290 10291 smtrace("port"); 10292 10293 switch (evt) { 10294 case OCS_EVT_ENTER: 10295 /* free SLI resource */ 10296 sli_resource_free(&hw->sli, SLI_RSRC_FCOE_VPI, sport->indicator); 10297 10298 /* free mailbox buffer */ 10299 if (data != NULL) { 10300 ocs_free(hw->os, data, SLI4_BMBX_SIZE); 10301 } 10302 10303 if (hw->callback.port != NULL) { 10304 hw->callback.port(hw->args.port, 10305 OCS_HW_PORT_ALLOC_FAIL, sport); 10306 } 10307 10308 /* If there is a pending free request, then handle it now */ 10309 if (sport->sm_free_req_pending) { 10310 ocs_sm_transition(ctx, __ocs_hw_port_free_unreg_vpi, NULL); 10311 } 10312 break; 10313 default: 10314 __ocs_hw_port_common(__func__, ctx, evt, data); 10315 break; 10316 } 10317 10318 return NULL; 10319 } 10320 10321 static void * 10322 __ocs_hw_port_alloc_read_sparm64(ocs_sm_ctx_t *ctx, ocs_sm_event_t evt, void *data) 10323 { 10324 ocs_sli_port_t *sport = ctx->app; 10325 ocs_hw_t *hw = sport->hw; 10326 uint8_t *payload = NULL; 10327 10328 smtrace("port"); 10329 10330 switch (evt) { 10331 case OCS_EVT_ENTER: 10332 /* allocate memory for the service parameters */ 10333 if (ocs_dma_alloc(hw->os, &sport->dma, 112, 4)) { 10334 ocs_log_err(hw->os, "Failed to allocate DMA memory\n"); 10335 ocs_sm_transition(ctx, __ocs_hw_port_done, data); 10336 break; 10337 } 10338 10339 if (0 == sli_cmd_read_sparm64(&hw->sli, data, SLI4_BMBX_SIZE, 10340 &sport->dma, sport->indicator)) { 10341 ocs_log_err(hw->os, "READ_SPARM64 allocation failure\n"); 10342 ocs_dma_free(hw->os, &sport->dma); 10343 ocs_sm_transition(ctx, __ocs_hw_port_done, data); 10344 break; 10345 } 10346 10347 if (ocs_hw_command(hw, data, OCS_CMD_NOWAIT, __ocs_hw_port_cb, sport)) { 10348 ocs_log_err(hw->os, "READ_SPARM64 command failure\n"); 10349 ocs_dma_free(hw->os, &sport->dma); 10350 ocs_sm_transition(ctx, __ocs_hw_port_done, data); 10351 break; 10352 } 10353 break; 10354 case OCS_EVT_RESPONSE: 10355 payload = sport->dma.virt; 10356 10357 ocs_display_sparams(sport->display_name, "sport sparm64", 0, NULL, payload); 10358 10359 ocs_memcpy(&sport->sli_wwpn, payload + SLI4_READ_SPARM64_WWPN_OFFSET, 10360 sizeof(sport->sli_wwpn)); 10361 ocs_memcpy(&sport->sli_wwnn, payload + SLI4_READ_SPARM64_WWNN_OFFSET, 10362 sizeof(sport->sli_wwnn)); 10363 10364 ocs_dma_free(hw->os, &sport->dma); 10365 ocs_sm_transition(ctx, __ocs_hw_port_alloc_init_vpi, data); 10366 break; 10367 case OCS_EVT_ERROR: 10368 ocs_dma_free(hw->os, &sport->dma); 10369 ocs_sm_transition(ctx, __ocs_hw_port_alloc_report_fail, data); 10370 break; 10371 case OCS_EVT_HW_PORT_REQ_FREE: 10372 /* Wait for attach response and then free */ 10373 sport->sm_free_req_pending = 1; 10374 break; 10375 case OCS_EVT_EXIT: 10376 break; 10377 default: 10378 __ocs_hw_port_common(__func__, ctx, evt, data); 10379 break; 10380 } 10381 10382 return NULL; 10383 } 10384 10385 static void * 10386 __ocs_hw_port_alloc_init(ocs_sm_ctx_t *ctx, ocs_sm_event_t evt, void *data) 10387 { 10388 ocs_sli_port_t *sport = ctx->app; 10389 10390 smtrace("port"); 10391 10392 switch (evt) { 10393 case OCS_EVT_ENTER: 10394 /* no-op */ 10395 break; 10396 case OCS_EVT_HW_PORT_ALLOC_OK: 10397 ocs_sm_transition(ctx, __ocs_hw_port_allocated, NULL); 10398 break; 10399 case OCS_EVT_HW_PORT_ALLOC_FAIL: 10400 ocs_sm_transition(ctx, __ocs_hw_port_alloc_report_fail, NULL); 10401 break; 10402 case OCS_EVT_HW_PORT_REQ_FREE: 10403 /* Wait for attach response and then free */ 10404 sport->sm_free_req_pending = 1; 10405 break; 10406 default: 10407 __ocs_hw_port_common(__func__, ctx, evt, data); 10408 break; 10409 } 10410 10411 return NULL; 10412 } 10413 10414 static void * 10415 __ocs_hw_port_alloc_init_vpi(ocs_sm_ctx_t *ctx, ocs_sm_event_t evt, void *data) 10416 { 10417 ocs_sli_port_t *sport = ctx->app; 10418 ocs_hw_t *hw = sport->hw; 10419 10420 smtrace("port"); 10421 10422 switch (evt) { 10423 case OCS_EVT_ENTER: 10424 /* If there is a pending free request, then handle it now */ 10425 if (sport->sm_free_req_pending) { 10426 ocs_sm_transition(ctx, __ocs_hw_port_freed, NULL); 10427 return NULL; 10428 } 10429 10430 /* TODO XXX transitioning to done only works if this is called 10431 * directly from ocs_hw_port_alloc BUT not if called from 10432 * read_sparm64. In the later case, we actually want to go 10433 * through report_ok/fail 10434 */ 10435 if (0 == sli_cmd_init_vpi(&hw->sli, data, SLI4_BMBX_SIZE, 10436 sport->indicator, sport->domain->indicator)) { 10437 ocs_log_err(hw->os, "INIT_VPI allocation failure\n"); 10438 ocs_sm_transition(ctx, __ocs_hw_port_done, data); 10439 break; 10440 } 10441 10442 if (ocs_hw_command(hw, data, OCS_CMD_NOWAIT, __ocs_hw_port_cb, sport)) { 10443 ocs_log_err(hw->os, "INIT_VPI command failure\n"); 10444 ocs_sm_transition(ctx, __ocs_hw_port_done, data); 10445 break; 10446 } 10447 break; 10448 case OCS_EVT_RESPONSE: 10449 ocs_sm_transition(ctx, __ocs_hw_port_allocated, data); 10450 break; 10451 case OCS_EVT_ERROR: 10452 ocs_sm_transition(ctx, __ocs_hw_port_alloc_report_fail, data); 10453 break; 10454 case OCS_EVT_HW_PORT_REQ_FREE: 10455 /* Wait for attach response and then free */ 10456 sport->sm_free_req_pending = 1; 10457 break; 10458 case OCS_EVT_EXIT: 10459 break; 10460 default: 10461 __ocs_hw_port_common(__func__, ctx, evt, data); 10462 break; 10463 } 10464 10465 return NULL; 10466 } 10467 10468 static int32_t 10469 __ocs_hw_port_cb(ocs_hw_t *hw, int32_t status, uint8_t *mqe, void *arg) 10470 { 10471 ocs_sli_port_t *sport = arg; 10472 sli4_mbox_command_header_t *hdr = (sli4_mbox_command_header_t *)mqe; 10473 ocs_sm_event_t evt; 10474 10475 if (status || hdr->status) { 10476 ocs_log_debug(hw->os, "bad status vpi=%#x st=%x hdr=%x\n", 10477 sport->indicator, status, hdr->status); 10478 evt = OCS_EVT_ERROR; 10479 } else { 10480 evt = OCS_EVT_RESPONSE; 10481 } 10482 10483 ocs_sm_post_event(&sport->ctx, evt, mqe); 10484 10485 return 0; 10486 } 10487 10488 static int32_t 10489 __ocs_hw_port_realloc_cb(ocs_hw_t *hw, int32_t status, uint8_t *mqe, void *arg) 10490 { 10491 ocs_sli_port_t *sport = arg; 10492 sli4_mbox_command_header_t *hdr = (sli4_mbox_command_header_t *)mqe; 10493 ocs_sm_event_t evt; 10494 uint8_t *mqecpy; 10495 10496 if (status || hdr->status) { 10497 ocs_log_debug(hw->os, "bad status vpi=%#x st=%x hdr=%x\n", 10498 sport->indicator, status, hdr->status); 10499 evt = OCS_EVT_ERROR; 10500 } else { 10501 evt = OCS_EVT_RESPONSE; 10502 } 10503 10504 /* 10505 * In this case we have to malloc a mailbox command buffer, as it is reused 10506 * in the state machine post event call, and eventually freed 10507 */ 10508 mqecpy = ocs_malloc(hw->os, SLI4_BMBX_SIZE, OCS_M_ZERO | OCS_M_NOWAIT); 10509 if (mqecpy == NULL) { 10510 ocs_log_err(hw->os, "malloc mqecpy failed\n"); 10511 return -1; 10512 } 10513 ocs_memcpy(mqecpy, mqe, SLI4_BMBX_SIZE); 10514 10515 ocs_sm_post_event(&sport->ctx, evt, mqecpy); 10516 10517 return 0; 10518 } 10519 10520 /*************************************************************************** 10521 * Domain state machine 10522 */ 10523 10524 static int32_t 10525 __ocs_hw_domain_common(const char *funcname, ocs_sm_ctx_t *ctx, ocs_sm_event_t evt, void *data) 10526 { 10527 ocs_domain_t *domain = ctx->app; 10528 ocs_hw_t *hw = domain->hw; 10529 10530 smtrace("domain"); 10531 10532 switch (evt) { 10533 case OCS_EVT_EXIT: 10534 /* ignore */ 10535 break; 10536 10537 default: 10538 ocs_log_test(hw->os, "%s %-20s not handled\n", funcname, ocs_sm_event_name(evt)); 10539 break; 10540 } 10541 10542 return 0; 10543 } 10544 10545 static void * 10546 __ocs_hw_domain_alloc_report_fail(ocs_sm_ctx_t *ctx, ocs_sm_event_t evt, void *data) 10547 { 10548 ocs_domain_t *domain = ctx->app; 10549 ocs_hw_t *hw = domain->hw; 10550 10551 smtrace("domain"); 10552 10553 switch (evt) { 10554 case OCS_EVT_ENTER: 10555 /* free command buffer */ 10556 if (data != NULL) { 10557 ocs_free(hw->os, data, SLI4_BMBX_SIZE); 10558 } 10559 /* free SLI resources */ 10560 sli_resource_free(&hw->sli, SLI_RSRC_FCOE_VFI, domain->indicator); 10561 /* TODO how to free FCFI (or do we at all)? */ 10562 10563 if (hw->callback.domain != NULL) { 10564 hw->callback.domain(hw->args.domain, 10565 OCS_HW_DOMAIN_ALLOC_FAIL, 10566 domain); 10567 } 10568 break; 10569 default: 10570 __ocs_hw_domain_common(__func__, ctx, evt, data); 10571 break; 10572 } 10573 10574 return NULL; 10575 } 10576 10577 static void * 10578 __ocs_hw_domain_attached(ocs_sm_ctx_t *ctx, ocs_sm_event_t evt, void *data) 10579 { 10580 ocs_domain_t *domain = ctx->app; 10581 ocs_hw_t *hw = domain->hw; 10582 10583 smtrace("domain"); 10584 10585 switch (evt) { 10586 case OCS_EVT_ENTER: 10587 /* free mailbox buffer and send alloc ok to physical sport */ 10588 ocs_free(hw->os, data, SLI4_BMBX_SIZE); 10589 ocs_sm_post_event(&domain->sport->ctx, OCS_EVT_HW_PORT_ATTACH_OK, NULL); 10590 10591 /* now inform registered callbacks */ 10592 if (hw->callback.domain != NULL) { 10593 hw->callback.domain(hw->args.domain, 10594 OCS_HW_DOMAIN_ATTACH_OK, 10595 domain); 10596 } 10597 break; 10598 case OCS_EVT_HW_DOMAIN_REQ_FREE: 10599 ocs_sm_transition(ctx, __ocs_hw_domain_free_unreg_vfi, NULL); 10600 break; 10601 default: 10602 __ocs_hw_domain_common(__func__, ctx, evt, data); 10603 break; 10604 } 10605 10606 return NULL; 10607 } 10608 10609 static void * 10610 __ocs_hw_domain_attach_report_fail(ocs_sm_ctx_t *ctx, ocs_sm_event_t evt, void *data) 10611 { 10612 ocs_domain_t *domain = ctx->app; 10613 ocs_hw_t *hw = domain->hw; 10614 10615 smtrace("domain"); 10616 10617 switch (evt) { 10618 case OCS_EVT_ENTER: 10619 if (data != NULL) { 10620 ocs_free(hw->os, data, SLI4_BMBX_SIZE); 10621 } 10622 /* free SLI resources */ 10623 sli_resource_free(&hw->sli, SLI_RSRC_FCOE_VFI, domain->indicator); 10624 /* TODO how to free FCFI (or do we at all)? */ 10625 10626 if (hw->callback.domain != NULL) { 10627 hw->callback.domain(hw->args.domain, 10628 OCS_HW_DOMAIN_ATTACH_FAIL, 10629 domain); 10630 } 10631 break; 10632 case OCS_EVT_EXIT: 10633 break; 10634 default: 10635 __ocs_hw_domain_common(__func__, ctx, evt, data); 10636 break; 10637 } 10638 10639 return NULL; 10640 } 10641 10642 static void * 10643 __ocs_hw_domain_attach_reg_vfi(ocs_sm_ctx_t *ctx, ocs_sm_event_t evt, void *data) 10644 { 10645 ocs_domain_t *domain = ctx->app; 10646 ocs_hw_t *hw = domain->hw; 10647 10648 smtrace("domain"); 10649 10650 switch (evt) { 10651 case OCS_EVT_ENTER: 10652 10653 ocs_display_sparams("", "reg vpi", 0, NULL, domain->dma.virt); 10654 10655 if (0 == sli_cmd_reg_vfi(&hw->sli, data, SLI4_BMBX_SIZE, domain)) { 10656 ocs_log_err(hw->os, "REG_VFI format failure\n"); 10657 ocs_sm_post_event(ctx, OCS_EVT_ERROR, NULL); 10658 break; 10659 } 10660 10661 if (ocs_hw_command(hw, data, OCS_CMD_NOWAIT, __ocs_hw_domain_cb, domain)) { 10662 ocs_log_err(hw->os, "REG_VFI command failure\n"); 10663 ocs_sm_post_event(ctx, OCS_EVT_ERROR, NULL); 10664 break; 10665 } 10666 break; 10667 case OCS_EVT_RESPONSE: 10668 ocs_sm_transition(ctx, __ocs_hw_domain_attached, data); 10669 break; 10670 case OCS_EVT_ERROR: 10671 ocs_sm_transition(ctx, __ocs_hw_domain_attach_report_fail, data); 10672 break; 10673 default: 10674 __ocs_hw_domain_common(__func__, ctx, evt, data); 10675 break; 10676 } 10677 10678 return NULL; 10679 } 10680 10681 static void * 10682 __ocs_hw_domain_allocated(ocs_sm_ctx_t *ctx, ocs_sm_event_t evt, void *data) 10683 { 10684 ocs_domain_t *domain = ctx->app; 10685 ocs_hw_t *hw = domain->hw; 10686 10687 smtrace("domain"); 10688 10689 switch (evt) { 10690 case OCS_EVT_ENTER: 10691 /* free mailbox buffer and send alloc ok to physical sport */ 10692 ocs_free(hw->os, data, SLI4_BMBX_SIZE); 10693 ocs_sm_post_event(&domain->sport->ctx, OCS_EVT_HW_PORT_ALLOC_OK, NULL); 10694 10695 ocs_hw_domain_add(hw, domain); 10696 10697 /* now inform registered callbacks */ 10698 if (hw->callback.domain != NULL) { 10699 hw->callback.domain(hw->args.domain, 10700 OCS_HW_DOMAIN_ALLOC_OK, 10701 domain); 10702 } 10703 break; 10704 case OCS_EVT_HW_DOMAIN_REQ_ATTACH: 10705 ocs_sm_transition(ctx, __ocs_hw_domain_attach_reg_vfi, data); 10706 break; 10707 case OCS_EVT_HW_DOMAIN_REQ_FREE: 10708 /* unreg_fcfi/vfi */ 10709 if (SLI4_IF_TYPE_BE3_SKH_PF == sli_get_if_type(&hw->sli)) { 10710 ocs_sm_transition(ctx, __ocs_hw_domain_free_unreg_fcfi, NULL); 10711 } else { 10712 ocs_sm_transition(ctx, __ocs_hw_domain_free_unreg_vfi, NULL); 10713 } 10714 break; 10715 default: 10716 __ocs_hw_domain_common(__func__, ctx, evt, data); 10717 break; 10718 } 10719 10720 return NULL; 10721 } 10722 10723 static void * 10724 __ocs_hw_domain_alloc_read_sparm64(ocs_sm_ctx_t *ctx, ocs_sm_event_t evt, void *data) 10725 { 10726 ocs_domain_t *domain = ctx->app; 10727 ocs_hw_t *hw = domain->hw; 10728 10729 smtrace("domain"); 10730 10731 switch (evt) { 10732 case OCS_EVT_ENTER: 10733 if (0 == sli_cmd_read_sparm64(&hw->sli, data, SLI4_BMBX_SIZE, 10734 &domain->dma, SLI4_READ_SPARM64_VPI_DEFAULT)) { 10735 ocs_log_err(hw->os, "READ_SPARM64 format failure\n"); 10736 ocs_sm_post_event(ctx, OCS_EVT_ERROR, NULL); 10737 break; 10738 } 10739 10740 if (ocs_hw_command(hw, data, OCS_CMD_NOWAIT, __ocs_hw_domain_cb, domain)) { 10741 ocs_log_err(hw->os, "READ_SPARM64 command failure\n"); 10742 ocs_sm_post_event(ctx, OCS_EVT_ERROR, NULL); 10743 break; 10744 } 10745 break; 10746 case OCS_EVT_EXIT: 10747 break; 10748 case OCS_EVT_RESPONSE: 10749 ocs_display_sparams(domain->display_name, "domain sparm64", 0, NULL, domain->dma.virt); 10750 10751 ocs_sm_transition(ctx, __ocs_hw_domain_allocated, data); 10752 break; 10753 case OCS_EVT_ERROR: 10754 ocs_sm_transition(ctx, __ocs_hw_domain_alloc_report_fail, data); 10755 break; 10756 default: 10757 __ocs_hw_domain_common(__func__, ctx, evt, data); 10758 break; 10759 } 10760 10761 return NULL; 10762 } 10763 10764 static void * 10765 __ocs_hw_domain_alloc_init_vfi(ocs_sm_ctx_t *ctx, ocs_sm_event_t evt, void *data) 10766 { 10767 ocs_domain_t *domain = ctx->app; 10768 ocs_sli_port_t *sport = domain->sport; 10769 ocs_hw_t *hw = domain->hw; 10770 10771 smtrace("domain"); 10772 10773 switch (evt) { 10774 case OCS_EVT_ENTER: 10775 if (0 == sli_cmd_init_vfi(&hw->sli, data, SLI4_BMBX_SIZE, domain->indicator, 10776 domain->fcf_indicator, sport->indicator)) { 10777 ocs_log_err(hw->os, "INIT_VFI format failure\n"); 10778 ocs_sm_post_event(ctx, OCS_EVT_ERROR, NULL); 10779 break; 10780 } 10781 if (ocs_hw_command(hw, data, OCS_CMD_NOWAIT, __ocs_hw_domain_cb, domain)) { 10782 ocs_log_err(hw->os, "INIT_VFI command failure\n"); 10783 ocs_sm_post_event(ctx, OCS_EVT_ERROR, NULL); 10784 break; 10785 } 10786 break; 10787 case OCS_EVT_EXIT: 10788 break; 10789 case OCS_EVT_RESPONSE: 10790 ocs_sm_transition(ctx, __ocs_hw_domain_alloc_read_sparm64, data); 10791 break; 10792 case OCS_EVT_ERROR: 10793 ocs_sm_transition(ctx, __ocs_hw_domain_alloc_report_fail, data); 10794 break; 10795 default: 10796 __ocs_hw_domain_common(__func__, ctx, evt, data); 10797 break; 10798 } 10799 10800 return NULL; 10801 } 10802 10803 static void * 10804 __ocs_hw_domain_alloc_reg_fcfi(ocs_sm_ctx_t *ctx, ocs_sm_event_t evt, void *data) 10805 { 10806 ocs_domain_t *domain = ctx->app; 10807 ocs_hw_t *hw = domain->hw; 10808 10809 smtrace("domain"); 10810 10811 switch (evt) { 10812 case OCS_EVT_ENTER: { 10813 sli4_cmd_rq_cfg_t rq_cfg[SLI4_CMD_REG_FCFI_NUM_RQ_CFG]; 10814 uint32_t i; 10815 10816 /* Set the filter match/mask values from hw's filter_def values */ 10817 for (i = 0; i < SLI4_CMD_REG_FCFI_NUM_RQ_CFG; i++) { 10818 rq_cfg[i].rq_id = 0xffff; 10819 rq_cfg[i].r_ctl_mask = (uint8_t) hw->config.filter_def[i]; 10820 rq_cfg[i].r_ctl_match = (uint8_t) (hw->config.filter_def[i] >> 8); 10821 rq_cfg[i].type_mask = (uint8_t) (hw->config.filter_def[i] >> 16); 10822 rq_cfg[i].type_match = (uint8_t) (hw->config.filter_def[i] >> 24); 10823 } 10824 10825 /* Set the rq_id for each, in order of RQ definition */ 10826 for (i = 0; i < hw->hw_rq_count; i++) { 10827 if (i >= ARRAY_SIZE(rq_cfg)) { 10828 ocs_log_warn(hw->os, "more RQs than REG_FCFI filter entries\n"); 10829 break; 10830 } 10831 rq_cfg[i].rq_id = hw->hw_rq[i]->hdr->id; 10832 } 10833 10834 if (!data) { 10835 ocs_sm_post_event(ctx, OCS_EVT_ERROR, NULL); 10836 break; 10837 } 10838 10839 if (hw->hw_mrq_count) { 10840 if (OCS_HW_RTN_SUCCESS != ocs_hw_config_mrq(hw, SLI4_CMD_REG_FCFI_SET_FCFI_MODE, 10841 domain->vlan_id, domain->fcf)) { 10842 ocs_log_err(hw->os, "REG_FCFI_MRQ format failure\n"); 10843 ocs_sm_post_event(ctx, OCS_EVT_ERROR, NULL); 10844 break; 10845 } 10846 10847 } else { 10848 if (0 == sli_cmd_reg_fcfi(&hw->sli, data, SLI4_BMBX_SIZE, domain->fcf, 10849 rq_cfg, domain->vlan_id)) { 10850 ocs_log_err(hw->os, "REG_FCFI format failure\n"); 10851 ocs_sm_post_event(ctx, OCS_EVT_ERROR, NULL); 10852 break; 10853 } 10854 } 10855 10856 if (ocs_hw_command(hw, data, OCS_CMD_NOWAIT, __ocs_hw_domain_cb, domain)) { 10857 ocs_log_err(hw->os, "REG_FCFI command failure\n"); 10858 ocs_sm_post_event(ctx, OCS_EVT_ERROR, NULL); 10859 break; 10860 } 10861 break; 10862 } 10863 case OCS_EVT_EXIT: 10864 break; 10865 case OCS_EVT_RESPONSE: 10866 if (!data) { 10867 ocs_sm_post_event(ctx, OCS_EVT_ERROR, NULL); 10868 break; 10869 } 10870 10871 domain->fcf_indicator = ((sli4_cmd_reg_fcfi_t *)data)->fcfi; 10872 10873 /* 10874 * IF_TYPE 0 devices do not support explicit VFI and VPI initialization 10875 * and instead rely on implicit initialization during VFI registration. 10876 * Short circuit normal processing here for those devices. 10877 */ 10878 if (SLI4_IF_TYPE_BE3_SKH_PF == sli_get_if_type(&hw->sli)) { 10879 ocs_sm_transition(ctx, __ocs_hw_domain_alloc_read_sparm64, data); 10880 } else { 10881 ocs_sm_transition(ctx, __ocs_hw_domain_alloc_init_vfi, data); 10882 } 10883 break; 10884 case OCS_EVT_ERROR: 10885 ocs_sm_transition(ctx, __ocs_hw_domain_alloc_report_fail, data); 10886 break; 10887 default: 10888 __ocs_hw_domain_common(__func__, ctx, evt, data); 10889 break; 10890 } 10891 10892 return NULL; 10893 } 10894 10895 static void * 10896 __ocs_hw_domain_init(ocs_sm_ctx_t *ctx, ocs_sm_event_t evt, void *data) 10897 { 10898 ocs_domain_t *domain = ctx->app; 10899 ocs_hw_t *hw = domain->hw; 10900 10901 smtrace("domain"); 10902 10903 switch (evt) { 10904 case OCS_EVT_ENTER: 10905 if (sli_get_medium(&hw->sli) == SLI_LINK_MEDIUM_FC) { 10906 /* 10907 * For FC, the HW alread registered a FCFI 10908 * Copy FCF information into the domain and jump to INIT_VFI 10909 */ 10910 domain->fcf_indicator = hw->fcf_indicator; 10911 ocs_sm_transition(&domain->sm, __ocs_hw_domain_alloc_init_vfi, data); 10912 } else { 10913 ocs_sm_transition(&domain->sm, __ocs_hw_domain_alloc_reg_fcfi, data); 10914 } 10915 break; 10916 default: 10917 __ocs_hw_domain_common(__func__, ctx, evt, data); 10918 break; 10919 } 10920 10921 return NULL; 10922 } 10923 10924 static void * 10925 __ocs_hw_domain_free_report_fail(ocs_sm_ctx_t *ctx, ocs_sm_event_t evt, void *data) 10926 { 10927 ocs_domain_t *domain = ctx->app; 10928 10929 smtrace("domain"); 10930 10931 switch (evt) { 10932 case OCS_EVT_ENTER: 10933 if (domain != NULL) { 10934 ocs_hw_t *hw = domain->hw; 10935 10936 ocs_hw_domain_del(hw, domain); 10937 10938 if (hw->callback.domain != NULL) { 10939 hw->callback.domain(hw->args.domain, 10940 OCS_HW_DOMAIN_FREE_FAIL, 10941 domain); 10942 } 10943 } 10944 10945 /* free command buffer */ 10946 if (data != NULL) { 10947 ocs_free(domain != NULL ? domain->hw->os : NULL, data, SLI4_BMBX_SIZE); 10948 } 10949 break; 10950 case OCS_EVT_EXIT: 10951 break; 10952 default: 10953 __ocs_hw_domain_common(__func__, ctx, evt, data); 10954 break; 10955 } 10956 10957 return NULL; 10958 } 10959 10960 static void * 10961 __ocs_hw_domain_freed(ocs_sm_ctx_t *ctx, ocs_sm_event_t evt, void *data) 10962 { 10963 ocs_domain_t *domain = ctx->app; 10964 10965 smtrace("domain"); 10966 10967 switch (evt) { 10968 case OCS_EVT_ENTER: 10969 /* Free DMA and mailbox buffer */ 10970 if (domain != NULL) { 10971 ocs_hw_t *hw = domain->hw; 10972 10973 /* free VFI resource */ 10974 sli_resource_free(&hw->sli, SLI_RSRC_FCOE_VFI, 10975 domain->indicator); 10976 10977 ocs_hw_domain_del(hw, domain); 10978 10979 /* inform registered callbacks */ 10980 if (hw->callback.domain != NULL) { 10981 hw->callback.domain(hw->args.domain, 10982 OCS_HW_DOMAIN_FREE_OK, 10983 domain); 10984 } 10985 } 10986 if (data != NULL) { 10987 ocs_free(NULL, data, SLI4_BMBX_SIZE); 10988 } 10989 break; 10990 case OCS_EVT_EXIT: 10991 break; 10992 default: 10993 __ocs_hw_domain_common(__func__, ctx, evt, data); 10994 break; 10995 } 10996 10997 return NULL; 10998 } 10999 11000 static void * 11001 __ocs_hw_domain_free_redisc_fcf(ocs_sm_ctx_t *ctx, ocs_sm_event_t evt, void *data) 11002 { 11003 ocs_domain_t *domain = ctx->app; 11004 ocs_hw_t *hw = domain->hw; 11005 11006 smtrace("domain"); 11007 11008 switch (evt) { 11009 case OCS_EVT_ENTER: 11010 /* if we're in the middle of a teardown, skip sending rediscover */ 11011 if (hw->state == OCS_HW_STATE_TEARDOWN_IN_PROGRESS) { 11012 ocs_sm_transition(ctx, __ocs_hw_domain_freed, data); 11013 break; 11014 } 11015 if (0 == sli_cmd_fcoe_rediscover_fcf(&hw->sli, data, SLI4_BMBX_SIZE, domain->fcf)) { 11016 ocs_log_err(hw->os, "REDISCOVER_FCF format failure\n"); 11017 ocs_sm_post_event(ctx, OCS_EVT_ERROR, NULL); 11018 break; 11019 } 11020 11021 if (ocs_hw_command(hw, data, OCS_CMD_NOWAIT, __ocs_hw_domain_cb, domain)) { 11022 ocs_log_err(hw->os, "REDISCOVER_FCF command failure\n"); 11023 ocs_sm_post_event(ctx, OCS_EVT_ERROR, NULL); 11024 } 11025 break; 11026 case OCS_EVT_RESPONSE: 11027 case OCS_EVT_ERROR: 11028 /* REDISCOVER_FCF can fail if none exist */ 11029 ocs_sm_transition(ctx, __ocs_hw_domain_freed, data); 11030 break; 11031 case OCS_EVT_EXIT: 11032 break; 11033 default: 11034 __ocs_hw_domain_common(__func__, ctx, evt, data); 11035 break; 11036 } 11037 11038 return NULL; 11039 } 11040 11041 static void * 11042 __ocs_hw_domain_free_unreg_fcfi(ocs_sm_ctx_t *ctx, ocs_sm_event_t evt, void *data) 11043 { 11044 ocs_domain_t *domain = ctx->app; 11045 ocs_hw_t *hw = domain->hw; 11046 11047 smtrace("domain"); 11048 11049 switch (evt) { 11050 case OCS_EVT_ENTER: 11051 if (data == NULL) { 11052 data = ocs_malloc(hw->os, SLI4_BMBX_SIZE, OCS_M_ZERO | OCS_M_NOWAIT); 11053 if (!data) { 11054 ocs_sm_post_event(ctx, OCS_EVT_ERROR, NULL); 11055 break; 11056 } 11057 } 11058 11059 if (0 == sli_cmd_unreg_fcfi(&hw->sli, data, SLI4_BMBX_SIZE, domain->fcf_indicator)) { 11060 ocs_log_err(hw->os, "UNREG_FCFI format failure\n"); 11061 ocs_free(hw->os, data, SLI4_BMBX_SIZE); 11062 ocs_sm_post_event(ctx, OCS_EVT_ERROR, NULL); 11063 break; 11064 } 11065 11066 if (ocs_hw_command(hw, data, OCS_CMD_NOWAIT, __ocs_hw_domain_cb, domain)) { 11067 ocs_log_err(hw->os, "UNREG_FCFI command failure\n"); 11068 ocs_free(hw->os, data, SLI4_BMBX_SIZE); 11069 ocs_sm_post_event(ctx, OCS_EVT_ERROR, NULL); 11070 break; 11071 } 11072 break; 11073 case OCS_EVT_RESPONSE: 11074 if (domain->req_rediscover_fcf) { 11075 domain->req_rediscover_fcf = FALSE; 11076 ocs_sm_transition(ctx, __ocs_hw_domain_free_redisc_fcf, data); 11077 } else { 11078 ocs_sm_transition(ctx, __ocs_hw_domain_freed, data); 11079 } 11080 break; 11081 case OCS_EVT_ERROR: 11082 ocs_sm_transition(ctx, __ocs_hw_domain_free_report_fail, data); 11083 break; 11084 case OCS_EVT_EXIT: 11085 break; 11086 default: 11087 __ocs_hw_domain_common(__func__, ctx, evt, data); 11088 break; 11089 } 11090 11091 return NULL; 11092 } 11093 11094 static void * 11095 __ocs_hw_domain_free_unreg_vfi(ocs_sm_ctx_t *ctx, ocs_sm_event_t evt, void *data) 11096 { 11097 ocs_domain_t *domain = ctx->app; 11098 ocs_hw_t *hw = domain->hw; 11099 uint8_t is_fc = FALSE; 11100 11101 smtrace("domain"); 11102 11103 is_fc = (sli_get_medium(&hw->sli) == SLI_LINK_MEDIUM_FC); 11104 11105 switch (evt) { 11106 case OCS_EVT_ENTER: 11107 if (data == NULL) { 11108 data = ocs_malloc(hw->os, SLI4_BMBX_SIZE, OCS_M_ZERO | OCS_M_NOWAIT); 11109 if (!data) { 11110 ocs_sm_post_event(ctx, OCS_EVT_ERROR, NULL); 11111 break; 11112 } 11113 } 11114 11115 if (0 == sli_cmd_unreg_vfi(&hw->sli, data, SLI4_BMBX_SIZE, domain, 11116 SLI4_UNREG_TYPE_DOMAIN)) { 11117 ocs_log_err(hw->os, "UNREG_VFI format failure\n"); 11118 ocs_free(hw->os, data, SLI4_BMBX_SIZE); 11119 ocs_sm_post_event(ctx, OCS_EVT_ERROR, NULL); 11120 break; 11121 } 11122 11123 if (ocs_hw_command(hw, data, OCS_CMD_NOWAIT, __ocs_hw_domain_cb, domain)) { 11124 ocs_log_err(hw->os, "UNREG_VFI command failure\n"); 11125 ocs_free(hw->os, data, SLI4_BMBX_SIZE); 11126 ocs_sm_post_event(ctx, OCS_EVT_ERROR, NULL); 11127 break; 11128 } 11129 break; 11130 case OCS_EVT_ERROR: 11131 if (is_fc) { 11132 ocs_sm_transition(ctx, __ocs_hw_domain_free_report_fail, data); 11133 } else { 11134 ocs_sm_transition(ctx, __ocs_hw_domain_free_unreg_fcfi, data); 11135 } 11136 break; 11137 case OCS_EVT_RESPONSE: 11138 if (is_fc) { 11139 ocs_sm_transition(ctx, __ocs_hw_domain_freed, data); 11140 } else { 11141 ocs_sm_transition(ctx, __ocs_hw_domain_free_unreg_fcfi, data); 11142 } 11143 break; 11144 default: 11145 __ocs_hw_domain_common(__func__, ctx, evt, data); 11146 break; 11147 } 11148 11149 return NULL; 11150 } 11151 11152 /* callback for domain alloc/attach/free */ 11153 static int32_t 11154 __ocs_hw_domain_cb(ocs_hw_t *hw, int32_t status, uint8_t *mqe, void *arg) 11155 { 11156 ocs_domain_t *domain = arg; 11157 sli4_mbox_command_header_t *hdr = (sli4_mbox_command_header_t *)mqe; 11158 ocs_sm_event_t evt; 11159 11160 if (status || hdr->status) { 11161 ocs_log_debug(hw->os, "bad status vfi=%#x st=%x hdr=%x\n", 11162 domain->indicator, status, hdr->status); 11163 evt = OCS_EVT_ERROR; 11164 } else { 11165 evt = OCS_EVT_RESPONSE; 11166 } 11167 11168 ocs_sm_post_event(&domain->sm, evt, mqe); 11169 11170 return 0; 11171 } 11172 11173 static int32_t 11174 target_wqe_timer_nop_cb(ocs_hw_t *hw, int32_t status, uint8_t *mqe, void *arg) 11175 { 11176 ocs_hw_io_t *io = NULL; 11177 ocs_hw_io_t *io_next = NULL; 11178 uint64_t ticks_current = ocs_get_os_ticks(); 11179 uint32_t sec_elapsed; 11180 ocs_hw_rtn_e rc; 11181 11182 sli4_mbox_command_header_t *hdr = (sli4_mbox_command_header_t *)mqe; 11183 11184 if (status || hdr->status) { 11185 ocs_log_debug(hw->os, "bad status st=%x hdr=%x\n", 11186 status, hdr->status); 11187 /* go ahead and proceed with wqe timer checks... */ 11188 } 11189 11190 /* loop through active WQE list and check for timeouts */ 11191 ocs_lock(&hw->io_lock); 11192 ocs_list_foreach_safe(&hw->io_timed_wqe, io, io_next) { 11193 sec_elapsed = ((ticks_current - io->submit_ticks) / ocs_get_os_tick_freq()); 11194 11195 /* 11196 * If elapsed time > timeout, abort it. No need to check type since 11197 * it wouldn't be on this list unless it was a target WQE 11198 */ 11199 if (sec_elapsed > io->tgt_wqe_timeout) { 11200 ocs_log_test(hw->os, "IO timeout xri=0x%x tag=0x%x type=%d\n", 11201 io->indicator, io->reqtag, io->type); 11202 11203 /* remove from active_wqe list so won't try to abort again */ 11204 ocs_list_remove(&hw->io_timed_wqe, io); 11205 11206 /* save status of "timed out" for when abort completes */ 11207 io->status_saved = 1; 11208 io->saved_status = SLI4_FC_WCQE_STATUS_TARGET_WQE_TIMEOUT; 11209 io->saved_ext = 0; 11210 io->saved_len = 0; 11211 11212 /* now abort outstanding IO */ 11213 rc = ocs_hw_io_abort(hw, io, FALSE, NULL, NULL); 11214 if (rc) { 11215 ocs_log_test(hw->os, 11216 "abort failed xri=%#x tag=%#x rc=%d\n", 11217 io->indicator, io->reqtag, rc); 11218 } 11219 } 11220 /* 11221 * need to go through entire list since each IO could have a 11222 * different timeout value 11223 */ 11224 } 11225 ocs_unlock(&hw->io_lock); 11226 11227 /* if we're not in the middle of shutting down, schedule next timer */ 11228 if (!hw->active_wqe_timer_shutdown) { 11229 ocs_setup_timer(hw->os, &hw->wqe_timer, target_wqe_timer_cb, hw, OCS_HW_WQ_TIMER_PERIOD_MS); 11230 } 11231 hw->in_active_wqe_timer = FALSE; 11232 return 0; 11233 } 11234 11235 static void 11236 target_wqe_timer_cb(void *arg) 11237 { 11238 ocs_hw_t *hw = (ocs_hw_t *)arg; 11239 11240 /* delete existing timer; will kick off new timer after checking wqe timeouts */ 11241 hw->in_active_wqe_timer = TRUE; 11242 ocs_del_timer(&hw->wqe_timer); 11243 11244 /* Forward timer callback to execute in the mailbox completion processing context */ 11245 if (ocs_hw_async_call(hw, target_wqe_timer_nop_cb, hw)) { 11246 ocs_log_test(hw->os, "ocs_hw_async_call failed\n"); 11247 } 11248 } 11249 11250 static void 11251 shutdown_target_wqe_timer(ocs_hw_t *hw) 11252 { 11253 uint32_t iters = 100; 11254 11255 if (hw->config.emulate_tgt_wqe_timeout) { 11256 /* request active wqe timer shutdown, then wait for it to complete */ 11257 hw->active_wqe_timer_shutdown = TRUE; 11258 11259 /* delete WQE timer and wait for timer handler to complete (if necessary) */ 11260 ocs_del_timer(&hw->wqe_timer); 11261 11262 /* now wait for timer handler to complete (if necessary) */ 11263 while (hw->in_active_wqe_timer && iters) { 11264 /* 11265 * if we happen to have just sent NOP mailbox command, make sure 11266 * completions are being processed 11267 */ 11268 ocs_hw_flush(hw); 11269 iters--; 11270 } 11271 11272 if (iters == 0) { 11273 ocs_log_test(hw->os, "Failed to shutdown active wqe timer\n"); 11274 } 11275 } 11276 } 11277 11278 /** 11279 * @brief Determine if HW IO is owned by the port. 11280 * 11281 * @par Description 11282 * Determines if the given HW IO has been posted to the chip. 11283 * 11284 * @param hw Hardware context allocated by the caller. 11285 * @param io HW IO. 11286 * 11287 * @return Returns TRUE if given HW IO is port-owned. 11288 */ 11289 uint8_t 11290 ocs_hw_is_io_port_owned(ocs_hw_t *hw, ocs_hw_io_t *io) 11291 { 11292 /* Check to see if this is a port owned XRI */ 11293 return io->is_port_owned; 11294 } 11295 11296 /** 11297 * @brief Return TRUE if exchange is port-owned. 11298 * 11299 * @par Description 11300 * Test to see if the xri is a port-owned xri. 11301 * 11302 * @param hw Hardware context. 11303 * @param xri Exchange indicator. 11304 * 11305 * @return Returns TRUE if XRI is a port owned XRI. 11306 */ 11307 11308 uint8_t 11309 ocs_hw_is_xri_port_owned(ocs_hw_t *hw, uint32_t xri) 11310 { 11311 ocs_hw_io_t *io = ocs_hw_io_lookup(hw, xri); 11312 return (io == NULL ? FALSE : io->is_port_owned); 11313 } 11314 11315 /** 11316 * @brief Returns an XRI from the port owned list to the host. 11317 * 11318 * @par Description 11319 * Used when the POST_XRI command fails as well as when the RELEASE_XRI completes. 11320 * 11321 * @param hw Hardware context. 11322 * @param xri_base The starting XRI number. 11323 * @param xri_count The number of XRIs to free from the base. 11324 */ 11325 static void 11326 ocs_hw_reclaim_xri(ocs_hw_t *hw, uint16_t xri_base, uint16_t xri_count) 11327 { 11328 ocs_hw_io_t *io; 11329 uint32_t i; 11330 11331 for (i = 0; i < xri_count; i++) { 11332 io = ocs_hw_io_lookup(hw, xri_base + i); 11333 11334 /* 11335 * if this is an auto xfer rdy XRI, then we need to release any 11336 * buffer attached to the XRI before moving the XRI back to the free pool. 11337 */ 11338 if (hw->auto_xfer_rdy_enabled) { 11339 ocs_hw_rqpair_auto_xfer_rdy_move_to_host(hw, io); 11340 } 11341 11342 ocs_lock(&hw->io_lock); 11343 ocs_list_remove(&hw->io_port_owned, io); 11344 io->is_port_owned = 0; 11345 ocs_list_add_tail(&hw->io_free, io); 11346 ocs_unlock(&hw->io_lock); 11347 } 11348 } 11349 11350 /** 11351 * @brief Called when the POST_XRI command completes. 11352 * 11353 * @par Description 11354 * Free the mailbox command buffer and reclaim the XRIs on failure. 11355 * 11356 * @param hw Hardware context. 11357 * @param status Status field from the mbox completion. 11358 * @param mqe Mailbox response structure. 11359 * @param arg Pointer to a callback function that signals the caller that the command is done. 11360 * 11361 * @return Returns 0. 11362 */ 11363 static int32_t 11364 ocs_hw_cb_post_xri(ocs_hw_t *hw, int32_t status, uint8_t *mqe, void *arg) 11365 { 11366 sli4_cmd_post_xri_t *post_xri = (sli4_cmd_post_xri_t*)mqe; 11367 11368 /* Reclaim the XRIs as host owned if the command fails */ 11369 if (status != 0) { 11370 ocs_log_debug(hw->os, "Status 0x%x for XRI base 0x%x, cnt =x%x\n", 11371 status, post_xri->xri_base, post_xri->xri_count); 11372 ocs_hw_reclaim_xri(hw, post_xri->xri_base, post_xri->xri_count); 11373 } 11374 11375 ocs_free(hw->os, mqe, SLI4_BMBX_SIZE); 11376 return 0; 11377 } 11378 11379 /** 11380 * @brief Issues a mailbox command to move XRIs from the host-controlled pool to the port. 11381 * 11382 * @param hw Hardware context. 11383 * @param xri_start The starting XRI to post. 11384 * @param num_to_post The number of XRIs to post. 11385 * 11386 * @return Returns OCS_HW_RTN_NO_MEMORY, OCS_HW_RTN_ERROR, or OCS_HW_RTN_SUCCESS. 11387 */ 11388 11389 static ocs_hw_rtn_e 11390 ocs_hw_post_xri(ocs_hw_t *hw, uint32_t xri_start, uint32_t num_to_post) 11391 { 11392 uint8_t *post_xri; 11393 ocs_hw_rtn_e rc = OCS_HW_RTN_ERROR; 11394 11395 /* Since we need to allocate for mailbox queue, just always allocate */ 11396 post_xri = ocs_malloc(hw->os, SLI4_BMBX_SIZE, OCS_M_NOWAIT); 11397 if (post_xri == NULL) { 11398 ocs_log_err(hw->os, "no buffer for command\n"); 11399 return OCS_HW_RTN_NO_MEMORY; 11400 } 11401 11402 /* Register the XRIs */ 11403 if (sli_cmd_post_xri(&hw->sli, post_xri, SLI4_BMBX_SIZE, 11404 xri_start, num_to_post)) { 11405 rc = ocs_hw_command(hw, post_xri, OCS_CMD_NOWAIT, ocs_hw_cb_post_xri, NULL); 11406 if (rc != OCS_HW_RTN_SUCCESS) { 11407 ocs_free(hw->os, post_xri, SLI4_BMBX_SIZE); 11408 ocs_log_err(hw->os, "post_xri failed\n"); 11409 } 11410 } 11411 return rc; 11412 } 11413 11414 /** 11415 * @brief Move XRIs from the host-controlled pool to the port. 11416 * 11417 * @par Description 11418 * Removes IOs from the free list and moves them to the port. 11419 * 11420 * @param hw Hardware context. 11421 * @param num_xri The number of XRIs being requested to move to the chip. 11422 * 11423 * @return Returns the number of XRIs that were moved. 11424 */ 11425 11426 uint32_t 11427 ocs_hw_xri_move_to_port_owned(ocs_hw_t *hw, uint32_t num_xri) 11428 { 11429 ocs_hw_io_t *io; 11430 uint32_t i; 11431 uint32_t num_posted = 0; 11432 11433 /* 11434 * Note: We cannot use ocs_hw_io_alloc() because that would place the 11435 * IO on the io_inuse list. We need to move from the io_free to 11436 * the io_port_owned list. 11437 */ 11438 ocs_lock(&hw->io_lock); 11439 11440 for (i = 0; i < num_xri; i++) { 11441 if (NULL != (io = ocs_list_remove_head(&hw->io_free))) { 11442 ocs_hw_rtn_e rc; 11443 11444 /* 11445 * if this is an auto xfer rdy XRI, then we need to attach a 11446 * buffer to the XRI before submitting it to the chip. If a 11447 * buffer is unavailable, then we cannot post it, so return it 11448 * to the free pool. 11449 */ 11450 if (hw->auto_xfer_rdy_enabled) { 11451 /* Note: uses the IO lock to get the auto xfer rdy buffer */ 11452 ocs_unlock(&hw->io_lock); 11453 rc = ocs_hw_rqpair_auto_xfer_rdy_move_to_port(hw, io); 11454 ocs_lock(&hw->io_lock); 11455 if (rc != OCS_HW_RTN_SUCCESS) { 11456 ocs_list_add_head(&hw->io_free, io); 11457 break; 11458 } 11459 } 11460 ocs_lock_init(hw->os, &io->axr_lock, "HW_axr_lock[%d]", io->indicator); 11461 io->is_port_owned = 1; 11462 ocs_list_add_tail(&hw->io_port_owned, io); 11463 11464 /* Post XRI */ 11465 if (ocs_hw_post_xri(hw, io->indicator, 1) != OCS_HW_RTN_SUCCESS ) { 11466 ocs_hw_reclaim_xri(hw, io->indicator, i); 11467 break; 11468 } 11469 num_posted++; 11470 } else { 11471 /* no more free XRIs */ 11472 break; 11473 } 11474 } 11475 ocs_unlock(&hw->io_lock); 11476 11477 return num_posted; 11478 } 11479 11480 /** 11481 * @brief Called when the RELEASE_XRI command completes. 11482 * 11483 * @par Description 11484 * Move the IOs back to the free pool on success. 11485 * 11486 * @param hw Hardware context. 11487 * @param status Status field from the mbox completion. 11488 * @param mqe Mailbox response structure. 11489 * @param arg Pointer to a callback function that signals the caller that the command is done. 11490 * 11491 * @return Returns 0. 11492 */ 11493 static int32_t 11494 ocs_hw_cb_release_xri(ocs_hw_t *hw, int32_t status, uint8_t *mqe, void *arg) 11495 { 11496 sli4_cmd_release_xri_t *release_xri = (sli4_cmd_release_xri_t*)mqe; 11497 uint8_t i; 11498 11499 /* Reclaim the XRIs as host owned if the command fails */ 11500 if (status != 0) { 11501 ocs_log_err(hw->os, "Status 0x%x\n", status); 11502 } else { 11503 for (i = 0; i < release_xri->released_xri_count; i++) { 11504 uint16_t xri = ((i & 1) == 0 ? release_xri->xri_tbl[i/2].xri_tag0 : 11505 release_xri->xri_tbl[i/2].xri_tag1); 11506 ocs_hw_reclaim_xri(hw, xri, 1); 11507 } 11508 } 11509 11510 ocs_free(hw->os, mqe, SLI4_BMBX_SIZE); 11511 return 0; 11512 } 11513 11514 /** 11515 * @brief Move XRIs from the port-controlled pool to the host. 11516 * 11517 * Requests XRIs from the FW to return to the host-owned pool. 11518 * 11519 * @param hw Hardware context. 11520 * @param num_xri The number of XRIs being requested to moved from the chip. 11521 * 11522 * @return Returns 0 for success, or a negative error code value for failure. 11523 */ 11524 11525 ocs_hw_rtn_e 11526 ocs_hw_xri_move_to_host_owned(ocs_hw_t *hw, uint8_t num_xri) 11527 { 11528 uint8_t *release_xri; 11529 ocs_hw_rtn_e rc = OCS_HW_RTN_ERROR; 11530 11531 /* non-local buffer required for mailbox queue */ 11532 release_xri = ocs_malloc(hw->os, SLI4_BMBX_SIZE, OCS_M_NOWAIT); 11533 if (release_xri == NULL) { 11534 ocs_log_err(hw->os, "no buffer for command\n"); 11535 return OCS_HW_RTN_NO_MEMORY; 11536 } 11537 11538 /* release the XRIs */ 11539 if (sli_cmd_release_xri(&hw->sli, release_xri, SLI4_BMBX_SIZE, num_xri)) { 11540 rc = ocs_hw_command(hw, release_xri, OCS_CMD_NOWAIT, ocs_hw_cb_release_xri, NULL); 11541 if (rc != OCS_HW_RTN_SUCCESS) { 11542 ocs_log_err(hw->os, "release_xri failed\n"); 11543 } 11544 } 11545 /* If we are polling or an error occurred, then free the mailbox buffer */ 11546 if (release_xri != NULL && rc != OCS_HW_RTN_SUCCESS) { 11547 ocs_free(hw->os, release_xri, SLI4_BMBX_SIZE); 11548 } 11549 return rc; 11550 } 11551 11552 /** 11553 * @brief Allocate an ocs_hw_rx_buffer_t array. 11554 * 11555 * @par Description 11556 * An ocs_hw_rx_buffer_t array is allocated, along with the required DMA memory. 11557 * 11558 * @param hw Pointer to HW object. 11559 * @param rqindex RQ index for this buffer. 11560 * @param count Count of buffers in array. 11561 * @param size Size of buffer. 11562 * 11563 * @return Returns the pointer to the allocated ocs_hw_rq_buffer_t array. 11564 */ 11565 static ocs_hw_rq_buffer_t * 11566 ocs_hw_rx_buffer_alloc(ocs_hw_t *hw, uint32_t rqindex, uint32_t count, uint32_t size) 11567 { 11568 ocs_t *ocs = hw->os; 11569 ocs_hw_rq_buffer_t *rq_buf = NULL; 11570 ocs_hw_rq_buffer_t *prq; 11571 uint32_t i; 11572 11573 if (count != 0) { 11574 rq_buf = ocs_malloc(hw->os, sizeof(*rq_buf) * count, OCS_M_NOWAIT | OCS_M_ZERO); 11575 if (rq_buf == NULL) { 11576 ocs_log_err(hw->os, "Failure to allocate unsolicited DMA trackers\n"); 11577 return NULL; 11578 } 11579 11580 for (i = 0, prq = rq_buf; i < count; i ++, prq++) { 11581 prq->rqindex = rqindex; 11582 if (ocs_dma_alloc(ocs, &prq->dma, size, OCS_MIN_DMA_ALIGNMENT)) { 11583 ocs_log_err(hw->os, "DMA allocation failed\n"); 11584 ocs_free(hw->os, rq_buf, sizeof(*rq_buf) * count); 11585 rq_buf = NULL; 11586 break; 11587 } 11588 } 11589 } 11590 return rq_buf; 11591 } 11592 11593 /** 11594 * @brief Free an ocs_hw_rx_buffer_t array. 11595 * 11596 * @par Description 11597 * The ocs_hw_rx_buffer_t array is freed, along with allocated DMA memory. 11598 * 11599 * @param hw Pointer to HW object. 11600 * @param rq_buf Pointer to ocs_hw_rx_buffer_t array. 11601 * @param count Count of buffers in array. 11602 * 11603 * @return None. 11604 */ 11605 static void 11606 ocs_hw_rx_buffer_free(ocs_hw_t *hw, ocs_hw_rq_buffer_t *rq_buf, uint32_t count) 11607 { 11608 ocs_t *ocs = hw->os; 11609 uint32_t i; 11610 ocs_hw_rq_buffer_t *prq; 11611 11612 if (rq_buf != NULL) { 11613 for (i = 0, prq = rq_buf; i < count; i++, prq++) { 11614 ocs_dma_free(ocs, &prq->dma); 11615 } 11616 ocs_free(hw->os, rq_buf, sizeof(*rq_buf) * count); 11617 } 11618 } 11619 11620 /** 11621 * @brief Allocate the RQ data buffers. 11622 * 11623 * @param hw Pointer to HW object. 11624 * 11625 * @return Returns 0 on success, or a non-zero value on failure. 11626 */ 11627 ocs_hw_rtn_e 11628 ocs_hw_rx_allocate(ocs_hw_t *hw) 11629 { 11630 ocs_t *ocs = hw->os; 11631 uint32_t i; 11632 int32_t rc = OCS_HW_RTN_SUCCESS; 11633 uint32_t rqindex = 0; 11634 hw_rq_t *rq; 11635 uint32_t hdr_size = OCS_HW_RQ_SIZE_HDR; 11636 uint32_t payload_size = hw->config.rq_default_buffer_size; 11637 11638 rqindex = 0; 11639 11640 for (i = 0; i < hw->hw_rq_count; i++) { 11641 rq = hw->hw_rq[i]; 11642 11643 /* Allocate header buffers */ 11644 rq->hdr_buf = ocs_hw_rx_buffer_alloc(hw, rqindex, rq->entry_count, hdr_size); 11645 if (rq->hdr_buf == NULL) { 11646 ocs_log_err(ocs, "ocs_hw_rx_buffer_alloc hdr_buf failed\n"); 11647 rc = OCS_HW_RTN_ERROR; 11648 break; 11649 } 11650 11651 ocs_log_debug(hw->os, "rq[%2d] rq_id %02d header %4d by %4d bytes\n", i, rq->hdr->id, 11652 rq->entry_count, hdr_size); 11653 11654 rqindex++; 11655 11656 /* Allocate payload buffers */ 11657 rq->payload_buf = ocs_hw_rx_buffer_alloc(hw, rqindex, rq->entry_count, payload_size); 11658 if (rq->payload_buf == NULL) { 11659 ocs_log_err(ocs, "ocs_hw_rx_buffer_alloc fb_buf failed\n"); 11660 rc = OCS_HW_RTN_ERROR; 11661 break; 11662 } 11663 ocs_log_debug(hw->os, "rq[%2d] rq_id %02d default %4d by %4d bytes\n", i, rq->data->id, 11664 rq->entry_count, payload_size); 11665 rqindex++; 11666 } 11667 11668 return rc ? OCS_HW_RTN_ERROR : OCS_HW_RTN_SUCCESS; 11669 } 11670 11671 /** 11672 * @brief Post the RQ data buffers to the chip. 11673 * 11674 * @param hw Pointer to HW object. 11675 * 11676 * @return Returns 0 on success, or a non-zero value on failure. 11677 */ 11678 ocs_hw_rtn_e 11679 ocs_hw_rx_post(ocs_hw_t *hw) 11680 { 11681 uint32_t i; 11682 uint32_t idx; 11683 uint32_t rq_idx; 11684 int32_t rc = 0; 11685 11686 /* 11687 * In RQ pair mode, we MUST post the header and payload buffer at the 11688 * same time. 11689 */ 11690 for (rq_idx = 0, idx = 0; rq_idx < hw->hw_rq_count; rq_idx++) { 11691 hw_rq_t *rq = hw->hw_rq[rq_idx]; 11692 11693 for (i = 0; i < rq->entry_count-1; i++) { 11694 ocs_hw_sequence_t *seq = ocs_array_get(hw->seq_pool, idx++); 11695 ocs_hw_assert(seq != NULL); 11696 11697 seq->header = &rq->hdr_buf[i]; 11698 11699 seq->payload = &rq->payload_buf[i]; 11700 11701 rc = ocs_hw_sequence_free(hw, seq); 11702 if (rc) { 11703 break; 11704 } 11705 } 11706 if (rc) { 11707 break; 11708 } 11709 } 11710 11711 return rc; 11712 } 11713 11714 /** 11715 * @brief Free the RQ data buffers. 11716 * 11717 * @param hw Pointer to HW object. 11718 * 11719 */ 11720 void 11721 ocs_hw_rx_free(ocs_hw_t *hw) 11722 { 11723 hw_rq_t *rq; 11724 uint32_t i; 11725 11726 /* Free hw_rq buffers */ 11727 for (i = 0; i < hw->hw_rq_count; i++) { 11728 rq = hw->hw_rq[i]; 11729 if (rq != NULL) { 11730 ocs_hw_rx_buffer_free(hw, rq->hdr_buf, rq->entry_count); 11731 rq->hdr_buf = NULL; 11732 ocs_hw_rx_buffer_free(hw, rq->payload_buf, rq->entry_count); 11733 rq->payload_buf = NULL; 11734 } 11735 } 11736 } 11737 11738 /** 11739 * @brief HW async call context structure. 11740 */ 11741 typedef struct { 11742 ocs_hw_async_cb_t callback; 11743 void *arg; 11744 uint8_t cmd[SLI4_BMBX_SIZE]; 11745 } ocs_hw_async_call_ctx_t; 11746 11747 /** 11748 * @brief HW async callback handler 11749 * 11750 * @par Description 11751 * This function is called when the NOP mailbox command completes. The callback stored 11752 * in the requesting context is invoked. 11753 * 11754 * @param hw Pointer to HW object. 11755 * @param status Completion status. 11756 * @param mqe Pointer to mailbox completion queue entry. 11757 * @param arg Caller-provided argument. 11758 * 11759 * @return None. 11760 */ 11761 static void 11762 ocs_hw_async_cb(ocs_hw_t *hw, int32_t status, uint8_t *mqe, void *arg) 11763 { 11764 ocs_hw_async_call_ctx_t *ctx = arg; 11765 11766 if (ctx != NULL) { 11767 if (ctx->callback != NULL) { 11768 (*ctx->callback)(hw, status, mqe, ctx->arg); 11769 } 11770 ocs_free(hw->os, ctx, sizeof(*ctx)); 11771 } 11772 } 11773 11774 /** 11775 * @brief Make an async callback using NOP mailbox command 11776 * 11777 * @par Description 11778 * Post a NOP mailbox command; the callback with argument is invoked upon completion 11779 * while in the event processing context. 11780 * 11781 * @param hw Pointer to HW object. 11782 * @param callback Pointer to callback function. 11783 * @param arg Caller-provided callback. 11784 * 11785 * @return Returns 0 on success, or a negative error code value on failure. 11786 */ 11787 int32_t 11788 ocs_hw_async_call(ocs_hw_t *hw, ocs_hw_async_cb_t callback, void *arg) 11789 { 11790 ocs_hw_async_call_ctx_t *ctx; 11791 11792 /* 11793 * Allocate a callback context (which includes the mailbox command buffer), we need 11794 * this to be persistent as the mailbox command submission may be queued and executed later 11795 * execution. 11796 */ 11797 ctx = ocs_malloc(hw->os, sizeof(*ctx), OCS_M_ZERO | OCS_M_NOWAIT); 11798 if (ctx == NULL) { 11799 ocs_log_err(hw->os, "failed to malloc async call context\n"); 11800 return OCS_HW_RTN_NO_MEMORY; 11801 } 11802 ctx->callback = callback; 11803 ctx->arg = arg; 11804 11805 /* Build and send a NOP mailbox command */ 11806 if (sli_cmd_common_nop(&hw->sli, ctx->cmd, sizeof(ctx->cmd), 0) == 0) { 11807 ocs_log_err(hw->os, "COMMON_NOP format failure\n"); 11808 ocs_free(hw->os, ctx, sizeof(*ctx)); 11809 return OCS_HW_RTN_ERROR; 11810 } 11811 11812 if (ocs_hw_command(hw, ctx->cmd, OCS_CMD_NOWAIT, ocs_hw_async_cb, ctx)) { 11813 ocs_log_err(hw->os, "COMMON_NOP command failure\n"); 11814 ocs_free(hw->os, ctx, sizeof(*ctx)); 11815 return OCS_HW_RTN_ERROR; 11816 } 11817 return OCS_HW_RTN_SUCCESS; 11818 } 11819 11820 /** 11821 * @brief Initialize the reqtag pool. 11822 * 11823 * @par Description 11824 * The WQ request tag pool is initialized. 11825 * 11826 * @param hw Pointer to HW object. 11827 * 11828 * @return Returns 0 on success, or a negative error code value on failure. 11829 */ 11830 ocs_hw_rtn_e 11831 ocs_hw_reqtag_init(ocs_hw_t *hw) 11832 { 11833 if (hw->wq_reqtag_pool == NULL) { 11834 hw->wq_reqtag_pool = ocs_pool_alloc(hw->os, sizeof(hw_wq_callback_t), 65536, TRUE); 11835 if (hw->wq_reqtag_pool == NULL) { 11836 ocs_log_err(hw->os, "ocs_pool_alloc hw_wq_callback_t failed\n"); 11837 return OCS_HW_RTN_NO_MEMORY; 11838 } 11839 } 11840 ocs_hw_reqtag_reset(hw); 11841 return OCS_HW_RTN_SUCCESS; 11842 } 11843 11844 /** 11845 * @brief Allocate a WQ request tag. 11846 * 11847 * Allocate and populate a WQ request tag from the WQ request tag pool. 11848 * 11849 * @param hw Pointer to HW object. 11850 * @param callback Callback function. 11851 * @param arg Pointer to callback argument. 11852 * 11853 * @return Returns pointer to allocated WQ request tag, or NULL if object cannot be allocated. 11854 */ 11855 hw_wq_callback_t * 11856 ocs_hw_reqtag_alloc(ocs_hw_t *hw, void (*callback)(void *arg, uint8_t *cqe, int32_t status), void *arg) 11857 { 11858 hw_wq_callback_t *wqcb; 11859 11860 ocs_hw_assert(callback != NULL); 11861 11862 wqcb = ocs_pool_get(hw->wq_reqtag_pool); 11863 if (wqcb != NULL) { 11864 ocs_hw_assert(wqcb->callback == NULL); 11865 wqcb->callback = callback; 11866 wqcb->arg = arg; 11867 } 11868 return wqcb; 11869 } 11870 11871 /** 11872 * @brief Free a WQ request tag. 11873 * 11874 * Free the passed in WQ request tag. 11875 * 11876 * @param hw Pointer to HW object. 11877 * @param wqcb Pointer to WQ request tag object to free. 11878 * 11879 * @return None. 11880 */ 11881 void 11882 ocs_hw_reqtag_free(ocs_hw_t *hw, hw_wq_callback_t *wqcb) 11883 { 11884 ocs_hw_assert(wqcb->callback != NULL); 11885 wqcb->callback = NULL; 11886 wqcb->arg = NULL; 11887 ocs_pool_put(hw->wq_reqtag_pool, wqcb); 11888 } 11889 11890 /** 11891 * @brief Return WQ request tag by index. 11892 * 11893 * @par Description 11894 * Return pointer to WQ request tag object given an index. 11895 * 11896 * @param hw Pointer to HW object. 11897 * @param instance_index Index of WQ request tag to return. 11898 * 11899 * @return Pointer to WQ request tag, or NULL. 11900 */ 11901 hw_wq_callback_t * 11902 ocs_hw_reqtag_get_instance(ocs_hw_t *hw, uint32_t instance_index) 11903 { 11904 hw_wq_callback_t *wqcb; 11905 11906 wqcb = ocs_pool_get_instance(hw->wq_reqtag_pool, instance_index); 11907 if (wqcb == NULL) { 11908 ocs_log_err(hw->os, "wqcb for instance %d is null\n", instance_index); 11909 } 11910 return wqcb; 11911 } 11912 11913 /** 11914 * @brief Reset the WQ request tag pool. 11915 * 11916 * @par Description 11917 * Reset the WQ request tag pool, returning all to the free list. 11918 * 11919 * @param hw pointer to HW object. 11920 * 11921 * @return None. 11922 */ 11923 void 11924 ocs_hw_reqtag_reset(ocs_hw_t *hw) 11925 { 11926 hw_wq_callback_t *wqcb; 11927 uint32_t i; 11928 11929 /* Remove all from freelist */ 11930 while(ocs_pool_get(hw->wq_reqtag_pool) != NULL) { 11931 ; 11932 } 11933 11934 /* Put them all back */ 11935 for (i = 0; ((wqcb = ocs_pool_get_instance(hw->wq_reqtag_pool, i)) != NULL); i++) { 11936 wqcb->instance_index = i; 11937 wqcb->callback = NULL; 11938 wqcb->arg = NULL; 11939 ocs_pool_put(hw->wq_reqtag_pool, wqcb); 11940 } 11941 } 11942 11943 /** 11944 * @brief Handle HW assertion 11945 * 11946 * HW assert, display diagnostic message, and abort. 11947 * 11948 * @param cond string describing failing assertion condition 11949 * @param filename file name 11950 * @param linenum line number 11951 * 11952 * @return none 11953 */ 11954 void 11955 _ocs_hw_assert(const char *cond, const char *filename, int linenum) 11956 { 11957 ocs_printf("%s(%d): HW assertion (%s) failed\n", filename, linenum, cond); 11958 ocs_abort(); 11959 /* no return */ 11960 } 11961 11962 /** 11963 * @brief Handle HW verify 11964 * 11965 * HW verify, display diagnostic message, dump stack and return. 11966 * 11967 * @param cond string describing failing verify condition 11968 * @param filename file name 11969 * @param linenum line number 11970 * 11971 * @return none 11972 */ 11973 void 11974 _ocs_hw_verify(const char *cond, const char *filename, int linenum) 11975 { 11976 ocs_printf("%s(%d): HW verify (%s) failed\n", filename, linenum, cond); 11977 ocs_print_stack(); 11978 } 11979 11980 /** 11981 * @brief Reque XRI 11982 * 11983 * @par Description 11984 * Reque XRI 11985 * 11986 * @param hw Pointer to HW object. 11987 * @param io Pointer to HW IO 11988 * 11989 * @return Return 0 if successful else returns -1 11990 */ 11991 int32_t 11992 ocs_hw_reque_xri( ocs_hw_t *hw, ocs_hw_io_t *io ) 11993 { 11994 int32_t rc = 0; 11995 11996 rc = ocs_hw_rqpair_auto_xfer_rdy_buffer_post(hw, io, 1); 11997 if (rc) { 11998 ocs_list_add_tail(&hw->io_port_dnrx, io); 11999 rc = -1; 12000 goto exit_ocs_hw_reque_xri; 12001 } 12002 12003 io->auto_xfer_rdy_dnrx = 0; 12004 io->type = OCS_HW_IO_DNRX_REQUEUE; 12005 if (sli_requeue_xri_wqe(&hw->sli, io->wqe.wqebuf, hw->sli.config.wqe_size, io->indicator, OCS_HW_REQUE_XRI_REGTAG, SLI4_CQ_DEFAULT)) { 12006 /* Clear buffer from XRI */ 12007 ocs_pool_put(hw->auto_xfer_rdy_buf_pool, io->axr_buf); 12008 io->axr_buf = NULL; 12009 12010 ocs_log_err(hw->os, "requeue_xri WQE error\n"); 12011 ocs_list_add_tail(&hw->io_port_dnrx, io); 12012 12013 rc = -1; 12014 goto exit_ocs_hw_reque_xri; 12015 } 12016 12017 if (io->wq == NULL) { 12018 io->wq = ocs_hw_queue_next_wq(hw, io); 12019 ocs_hw_assert(io->wq != NULL); 12020 } 12021 12022 /* 12023 * Add IO to active io wqe list before submitting, in case the 12024 * wcqe processing preempts this thread. 12025 */ 12026 OCS_STAT(hw->tcmd_wq_submit[io->wq->instance]++); 12027 OCS_STAT(io->wq->use_count++); 12028 12029 rc = hw_wq_write(io->wq, &io->wqe); 12030 if (rc < 0) { 12031 ocs_log_err(hw->os, "sli_queue_write reque xri failed: %d\n", rc); 12032 rc = -1; 12033 } 12034 12035 exit_ocs_hw_reque_xri: 12036 return 0; 12037 } 12038 12039 uint32_t 12040 ocs_hw_get_def_wwn(ocs_t *ocs, uint32_t chan, uint64_t *wwpn, uint64_t *wwnn) 12041 { 12042 sli4_t *sli4 = &ocs->hw.sli; 12043 ocs_dma_t dma; 12044 uint8_t *payload = NULL; 12045 12046 int indicator = sli4->config.extent[SLI_RSRC_FCOE_VPI].base[0] + chan; 12047 12048 /* allocate memory for the service parameters */ 12049 if (ocs_dma_alloc(ocs, &dma, 112, 4)) { 12050 ocs_log_err(ocs, "Failed to allocate DMA memory\n"); 12051 return 1; 12052 } 12053 12054 if (0 == sli_cmd_read_sparm64(sli4, sli4->bmbx.virt, SLI4_BMBX_SIZE, 12055 &dma, indicator)) { 12056 ocs_log_err(ocs, "READ_SPARM64 allocation failure\n"); 12057 ocs_dma_free(ocs, &dma); 12058 return 1; 12059 } 12060 12061 if (sli_bmbx_command(sli4)) { 12062 ocs_log_err(ocs, "READ_SPARM64 command failure\n"); 12063 ocs_dma_free(ocs, &dma); 12064 return 1; 12065 } 12066 12067 payload = dma.virt; 12068 ocs_memcpy(wwpn, payload + SLI4_READ_SPARM64_WWPN_OFFSET, sizeof(*wwpn)); 12069 ocs_memcpy(wwnn, payload + SLI4_READ_SPARM64_WWNN_OFFSET, sizeof(*wwnn)); 12070 ocs_dma_free(ocs, &dma); 12071 return 0; 12072 } 12073 12074 /** 12075 * @page fc_hw_api_overview HW APIs 12076 * - @ref devInitShutdown 12077 * - @ref domain 12078 * - @ref port 12079 * - @ref node 12080 * - @ref io 12081 * - @ref interrupt 12082 * 12083 * <div class="overview"> 12084 * The Hardware Abstraction Layer (HW) insulates the higher-level code from the SLI-4 12085 * message details, but the higher level code must still manage domains, ports, 12086 * IT nexuses, and IOs. The HW API is designed to help the higher level manage 12087 * these objects.<br><br> 12088 * 12089 * The HW uses function callbacks to notify the higher-level code of events 12090 * that are received from the chip. There are currently three types of 12091 * functions that may be registered: 12092 * 12093 * <ul><li>domain – This function is called whenever a domain event is generated 12094 * within the HW. Examples include a new FCF is discovered, a connection 12095 * to a domain is disrupted, and allocation callbacks.</li> 12096 * <li>unsolicited – This function is called whenever new data is received in 12097 * the SLI-4 receive queue.</li> 12098 * <li>rnode – This function is called for remote node events, such as attach status 12099 * and allocation callbacks.</li></ul> 12100 * 12101 * Upper layer functions may be registered by using the ocs_hw_callback() function. 12102 * 12103 * <img src="elx_fc_hw.jpg" alt="FC/FCoE HW" title="FC/FCoE HW" align="right"/> 12104 * <h2>FC/FCoE HW API</h2> 12105 * The FC/FCoE HW component builds upon the SLI-4 component to establish a flexible 12106 * interface for creating the necessary common objects and sending I/Os. It may be used 12107 * “as is” in customer implementations or it can serve as an example of typical interactions 12108 * between a driver and the SLI-4 hardware. The broad categories of functionality include: 12109 * 12110 * <ul><li>Setting-up and tearing-down of the HW.</li> 12111 * <li>Allocating and using the common objects (SLI Port, domain, remote node).</li> 12112 * <li>Sending and receiving I/Os.</li></ul> 12113 * 12114 * <h3>HW Setup</h3> 12115 * To set up the HW: 12116 * 12117 * <ol> 12118 * <li>Set up the HW object using ocs_hw_setup().<br> 12119 * This step performs a basic configuration of the SLI-4 component and the HW to 12120 * enable querying the hardware for its capabilities. At this stage, the HW is not 12121 * capable of general operations (such as, receiving events or sending I/Os).</li><br><br> 12122 * <li>Configure the HW according to the driver requirements.<br> 12123 * The HW provides functions to discover hardware capabilities (ocs_hw_get()), as 12124 * well as configures the amount of resources required (ocs_hw_set()). The driver 12125 * must also register callback functions (ocs_hw_callback()) to receive notification of 12126 * various asynchronous events.<br><br> 12127 * @b Note: Once configured, the driver must initialize the HW (ocs_hw_init()). This 12128 * step creates the underlying queues, commits resources to the hardware, and 12129 * prepares the hardware for operation. While the hardware is operational, the 12130 * port is not online, and cannot send or receive data.</li><br><br> 12131 * <br><br> 12132 * <li>Finally, the driver can bring the port online (ocs_hw_port_control()).<br> 12133 * When the link comes up, the HW determines if a domain is present and notifies the 12134 * driver using the domain callback function. This is the starting point of the driver's 12135 * interaction with the common objects.<br><br> 12136 * @b Note: For FCoE, there may be more than one domain available and, therefore, 12137 * more than one callback.</li> 12138 * </ol> 12139 * 12140 * <h3>Allocating and Using Common Objects</h3> 12141 * Common objects provide a mechanism through which the various OneCore Storage 12142 * driver components share and track information. These data structures are primarily 12143 * used to track SLI component information but can be extended by other components, if 12144 * needed. The main objects are: 12145 * 12146 * <ul><li>DMA – the ocs_dma_t object describes a memory region suitable for direct 12147 * memory access (DMA) transactions.</li> 12148 * <li>SCSI domain – the ocs_domain_t object represents the SCSI domain, including 12149 * any infrastructure devices such as FC switches and FC forwarders. The domain 12150 * object contains both an FCFI and a VFI.</li> 12151 * <li>SLI Port (sport) – the ocs_sli_port_t object represents the connection between 12152 * the driver and the SCSI domain. The SLI Port object contains a VPI.</li> 12153 * <li>Remote node – the ocs_remote_node_t represents a connection between the SLI 12154 * Port and another device in the SCSI domain. The node object contains an RPI.</li></ul> 12155 * 12156 * Before the driver can send I/Os, it must allocate the SCSI domain, SLI Port, and remote 12157 * node common objects and establish the connections between them. The goal is to 12158 * connect the driver to the SCSI domain to exchange I/Os with other devices. These 12159 * common object connections are shown in the following figure, FC Driver Common Objects: 12160 * <img src="elx_fc_common_objects.jpg" 12161 * alt="FC Driver Common Objects" title="FC Driver Common Objects" align="center"/> 12162 * 12163 * The first step is to create a connection to the domain by allocating an SLI Port object. 12164 * The SLI Port object represents a particular FC ID and must be initialized with one. With 12165 * the SLI Port object, the driver can discover the available SCSI domain(s). On identifying 12166 * a domain, the driver allocates a domain object and attaches to it using the previous SLI 12167 * port object.<br><br> 12168 * 12169 * @b Note: In some cases, the driver may need to negotiate service parameters (that is, 12170 * FLOGI) with the domain before attaching.<br><br> 12171 * 12172 * Once attached to the domain, the driver can discover and attach to other devices 12173 * (remote nodes). The exact discovery method depends on the driver, but it typically 12174 * includes using a position map, querying the fabric name server, or an out-of-band 12175 * method. In most cases, it is necessary to log in with devices before performing I/Os. 12176 * Prior to sending login-related ELS commands (ocs_hw_srrs_send()), the driver must 12177 * allocate a remote node object (ocs_hw_node_alloc()). If the login negotiation is 12178 * successful, the driver must attach the nodes (ocs_hw_node_attach()) to the SLI Port 12179 * before exchanging FCP I/O.<br><br> 12180 * 12181 * @b Note: The HW manages both the well known fabric address and the name server as 12182 * nodes in the domain. Therefore, the driver must allocate node objects prior to 12183 * communicating with either of these entities. 12184 * 12185 * <h3>Sending and Receiving I/Os</h3> 12186 * The HW provides separate interfaces for sending BLS/ ELS/ FC-CT and FCP, but the 12187 * commands are conceptually similar. Since the commands complete asynchronously, 12188 * the caller must provide a HW I/O object that maintains the I/O state, as well as 12189 * provide a callback function. The driver may use the same callback function for all I/O 12190 * operations, but each operation must use a unique HW I/O object. In the SLI-4 12191 * architecture, there is a direct association between the HW I/O object and the SGL used 12192 * to describe the data. Therefore, a driver typically performs the following operations: 12193 * 12194 * <ul><li>Allocates a HW I/O object (ocs_hw_io_alloc()).</li> 12195 * <li>Formats the SGL, specifying both the HW I/O object and the SGL. 12196 * (ocs_hw_io_init_sges() and ocs_hw_io_add_sge()).</li> 12197 * <li>Sends the HW I/O (ocs_hw_io_send()).</li></ul> 12198 * 12199 * <h3>HW Tear Down</h3> 12200 * To tear-down the HW: 12201 * 12202 * <ol><li>Take the port offline (ocs_hw_port_control()) to prevent receiving further 12203 * data andevents.</li> 12204 * <li>Destroy the HW object (ocs_hw_teardown()).</li> 12205 * <li>Free any memory used by the HW, such as buffers for unsolicited data.</li></ol> 12206 * <br> 12207 * </div><!-- overview --> 12208 * 12209 */ 12210 12211 /** 12212 * This contains all hw runtime workaround code. Based on the asic type, 12213 * asic revision, and range of fw revisions, a particular workaround may be enabled. 12214 * 12215 * A workaround may consist of overriding a particular HW/SLI4 value that was initialized 12216 * during ocs_hw_setup() (for example the MAX_QUEUE overrides for mis-reported queue 12217 * sizes). Or if required, elements of the ocs_hw_workaround_t structure may be set to 12218 * control specific runtime behavior. 12219 * 12220 * It is intended that the controls in ocs_hw_workaround_t be defined functionally. So we 12221 * would have the driver look like: "if (hw->workaround.enable_xxx) then ...", rather than 12222 * what we might previously see as "if this is a BE3, then do xxx" 12223 * 12224 */ 12225 12226 #define HW_FWREV_ZERO (0ull) 12227 #define HW_FWREV_MAX (~0ull) 12228 12229 #define SLI4_ASIC_TYPE_ANY 0 12230 #define SLI4_ASIC_REV_ANY 0 12231 12232 /** 12233 * @brief Internal definition of workarounds 12234 */ 12235 12236 typedef enum { 12237 HW_WORKAROUND_TEST = 1, 12238 HW_WORKAROUND_MAX_QUEUE, /**< Limits all queues */ 12239 HW_WORKAROUND_MAX_RQ, /**< Limits only the RQ */ 12240 HW_WORKAROUND_RETAIN_TSEND_IO_LENGTH, 12241 HW_WORKAROUND_WQE_COUNT_METHOD, 12242 HW_WORKAROUND_RQE_COUNT_METHOD, 12243 HW_WORKAROUND_USE_UNREGISTERD_RPI, 12244 HW_WORKAROUND_DISABLE_AR_TGT_DIF, /**< Disable of auto-response target DIF */ 12245 HW_WORKAROUND_DISABLE_SET_DUMP_LOC, 12246 HW_WORKAROUND_USE_DIF_QUARANTINE, 12247 HW_WORKAROUND_USE_DIF_SEC_XRI, /**< Use secondary xri for multiple data phases */ 12248 HW_WORKAROUND_OVERRIDE_FCFI_IN_SRB, /**< FCFI reported in SRB not correct, use "first" registered domain */ 12249 HW_WORKAROUND_FW_VERSION_TOO_LOW, /**< The FW version is not the min version supported by this driver */ 12250 HW_WORKAROUND_SGLC_MISREPORTED, /**< Chip supports SGL Chaining but SGLC is not set in SLI4_PARAMS */ 12251 HW_WORKAROUND_IGNORE_SEND_FRAME_CAPABLE, /**< Don't use SEND_FRAME capable if FW version is too old */ 12252 } hw_workaround_e; 12253 12254 /** 12255 * @brief Internal workaround structure instance 12256 */ 12257 12258 typedef struct { 12259 sli4_asic_type_e asic_type; 12260 sli4_asic_rev_e asic_rev; 12261 uint64_t fwrev_low; 12262 uint64_t fwrev_high; 12263 12264 hw_workaround_e workaround; 12265 uint32_t value; 12266 } hw_workaround_t; 12267 12268 static hw_workaround_t hw_workarounds[] = { 12269 {SLI4_ASIC_TYPE_ANY, SLI4_ASIC_REV_ANY, HW_FWREV_ZERO, HW_FWREV_MAX, 12270 HW_WORKAROUND_TEST, 999}, 12271 12272 /* Bug: 127585: if_type == 2 returns 0 for total length placed on 12273 * FCP_TSEND64_WQE completions. Note, original driver code enables this 12274 * workaround for all asic types 12275 */ 12276 {SLI4_ASIC_TYPE_ANY, SLI4_ASIC_REV_ANY, HW_FWREV_ZERO, HW_FWREV_MAX, 12277 HW_WORKAROUND_RETAIN_TSEND_IO_LENGTH, 0}, 12278 12279 /* Bug: unknown, Lancer A0 has mis-reported max queue depth */ 12280 {SLI4_ASIC_TYPE_LANCER, SLI4_ASIC_REV_A0, HW_FWREV_ZERO, HW_FWREV_MAX, 12281 HW_WORKAROUND_MAX_QUEUE, 2048}, 12282 12283 /* Bug: 143399, BE3 has mis-reported max RQ queue depth */ 12284 {SLI4_ASIC_TYPE_BE3, SLI4_ASIC_REV_ANY, HW_FWREV_ZERO, HW_FWREV(4,6,293,0), 12285 HW_WORKAROUND_MAX_RQ, 2048}, 12286 12287 /* Bug: 143399, skyhawk has mis-reported max RQ queue depth */ 12288 {SLI4_ASIC_TYPE_SKYHAWK, SLI4_ASIC_REV_ANY, HW_FWREV_ZERO, HW_FWREV(10,0,594,0), 12289 HW_WORKAROUND_MAX_RQ, 2048}, 12290 12291 /* Bug: 103487, BE3 before f/w 4.2.314.0 has mis-reported WQE count method */ 12292 {SLI4_ASIC_TYPE_BE3, SLI4_ASIC_REV_ANY, HW_FWREV_ZERO, HW_FWREV(4,2,314,0), 12293 HW_WORKAROUND_WQE_COUNT_METHOD, 1}, 12294 12295 /* Bug: 103487, BE3 before f/w 4.2.314.0 has mis-reported RQE count method */ 12296 {SLI4_ASIC_TYPE_BE3, SLI4_ASIC_REV_ANY, HW_FWREV_ZERO, HW_FWREV(4,2,314,0), 12297 HW_WORKAROUND_RQE_COUNT_METHOD, 1}, 12298 12299 /* Bug: 142968, BE3 UE with RPI == 0xffff */ 12300 {SLI4_ASIC_TYPE_BE3, SLI4_ASIC_REV_ANY, HW_FWREV_ZERO, HW_FWREV_MAX, 12301 HW_WORKAROUND_USE_UNREGISTERD_RPI, 0}, 12302 12303 /* Bug: unknown, Skyhawk won't support auto-response on target T10-PI */ 12304 {SLI4_ASIC_TYPE_SKYHAWK, SLI4_ASIC_REV_ANY, HW_FWREV_ZERO, HW_FWREV_MAX, 12305 HW_WORKAROUND_DISABLE_AR_TGT_DIF, 0}, 12306 12307 {SLI4_ASIC_TYPE_LANCER, SLI4_ASIC_REV_ANY, HW_FWREV_ZERO, HW_FWREV(1,1,65,0), 12308 HW_WORKAROUND_DISABLE_SET_DUMP_LOC, 0}, 12309 12310 /* Bug: 160124, Skyhawk quarantine DIF XRIs */ 12311 {SLI4_ASIC_TYPE_SKYHAWK, SLI4_ASIC_REV_ANY, HW_FWREV_ZERO, HW_FWREV_MAX, 12312 HW_WORKAROUND_USE_DIF_QUARANTINE, 0}, 12313 12314 /* Bug: 161832, Skyhawk use secondary XRI for multiple data phase TRECV */ 12315 {SLI4_ASIC_TYPE_SKYHAWK, SLI4_ASIC_REV_ANY, HW_FWREV_ZERO, HW_FWREV_MAX, 12316 HW_WORKAROUND_USE_DIF_SEC_XRI, 0}, 12317 12318 /* Bug: xxxxxx, FCFI reported in SRB not corrrect */ 12319 {SLI4_ASIC_TYPE_LANCER, SLI4_ASIC_REV_ANY, HW_FWREV_ZERO, HW_FWREV_MAX, 12320 HW_WORKAROUND_OVERRIDE_FCFI_IN_SRB, 0}, 12321 #if 0 12322 /* Bug: 165642, FW version check for driver */ 12323 {SLI4_ASIC_TYPE_LANCER, SLI4_ASIC_REV_ANY, HW_FWREV_ZERO, HW_FWREV_1(OCS_MIN_FW_VER_LANCER), 12324 HW_WORKAROUND_FW_VERSION_TOO_LOW, 0}, 12325 #endif 12326 {SLI4_ASIC_TYPE_SKYHAWK, SLI4_ASIC_REV_ANY, HW_FWREV_ZERO, HW_FWREV_1(OCS_MIN_FW_VER_SKYHAWK), 12327 HW_WORKAROUND_FW_VERSION_TOO_LOW, 0}, 12328 12329 /* Bug 177061, Lancer FW does not set the SGLC bit */ 12330 {SLI4_ASIC_TYPE_LANCER, SLI4_ASIC_REV_ANY, HW_FWREV_ZERO, HW_FWREV_MAX, 12331 HW_WORKAROUND_SGLC_MISREPORTED, 0}, 12332 12333 /* BZ 181208/183914, enable this workaround for ALL revisions */ 12334 {SLI4_ASIC_TYPE_ANY, SLI4_ASIC_REV_ANY, HW_FWREV_ZERO, HW_FWREV_MAX, 12335 HW_WORKAROUND_IGNORE_SEND_FRAME_CAPABLE, 0}, 12336 }; 12337 12338 /** 12339 * @brief Function prototypes 12340 */ 12341 12342 static int32_t ocs_hw_workaround_match(ocs_hw_t *hw, hw_workaround_t *w); 12343 12344 /** 12345 * @brief Parse the firmware version (name) 12346 * 12347 * Parse a string of the form a.b.c.d, returning a uint64_t packed as defined 12348 * by the HW_FWREV() macro 12349 * 12350 * @param fwrev_string pointer to the firmware string 12351 * 12352 * @return packed firmware revision value 12353 */ 12354 12355 static uint64_t 12356 parse_fw_version(const char *fwrev_string) 12357 { 12358 int v[4] = {0}; 12359 const char *p; 12360 int i; 12361 12362 for (p = fwrev_string, i = 0; *p && (i < 4); i ++) { 12363 v[i] = ocs_strtoul(p, 0, 0); 12364 while(*p && *p != '.') { 12365 p ++; 12366 } 12367 if (*p) { 12368 p ++; 12369 } 12370 } 12371 12372 /* Special case for bootleg releases with f/w rev 0.0.9999.0, set to max value */ 12373 if (v[2] == 9999) { 12374 return HW_FWREV_MAX; 12375 } else { 12376 return HW_FWREV(v[0], v[1], v[2], v[3]); 12377 } 12378 } 12379 12380 /** 12381 * @brief Test for a workaround match 12382 * 12383 * Looks at the asic type, asic revision, and fw revision, and returns TRUE if match. 12384 * 12385 * @param hw Pointer to the HW structure 12386 * @param w Pointer to a workaround structure entry 12387 * 12388 * @return Return TRUE for a match 12389 */ 12390 12391 static int32_t 12392 ocs_hw_workaround_match(ocs_hw_t *hw, hw_workaround_t *w) 12393 { 12394 return (((w->asic_type == SLI4_ASIC_TYPE_ANY) || (w->asic_type == hw->sli.asic_type)) && 12395 ((w->asic_rev == SLI4_ASIC_REV_ANY) || (w->asic_rev == hw->sli.asic_rev)) && 12396 (w->fwrev_low <= hw->workaround.fwrev) && 12397 ((w->fwrev_high == HW_FWREV_MAX) || (hw->workaround.fwrev < w->fwrev_high))); 12398 } 12399 12400 /** 12401 * @brief Setup HW runtime workarounds 12402 * 12403 * The function is called at the end of ocs_hw_setup() to setup any runtime workarounds 12404 * based on the HW/SLI setup. 12405 * 12406 * @param hw Pointer to HW structure 12407 * 12408 * @return none 12409 */ 12410 12411 void 12412 ocs_hw_workaround_setup(struct ocs_hw_s *hw) 12413 { 12414 hw_workaround_t *w; 12415 sli4_t *sli4 = &hw->sli; 12416 uint32_t i; 12417 12418 /* Initialize the workaround settings */ 12419 ocs_memset(&hw->workaround, 0, sizeof(hw->workaround)); 12420 12421 /* If hw_war_version is non-null, then its a value that was set by a module parameter 12422 * (sorry for the break in abstraction, but workarounds are ... well, workarounds) 12423 */ 12424 12425 if (hw->hw_war_version) { 12426 hw->workaround.fwrev = parse_fw_version(hw->hw_war_version); 12427 } else { 12428 hw->workaround.fwrev = parse_fw_version((char*) sli4->config.fw_name[0]); 12429 } 12430 12431 /* Walk the workaround list, if a match is found, then handle it */ 12432 for (i = 0, w = hw_workarounds; i < ARRAY_SIZE(hw_workarounds); i++, w++) { 12433 if (ocs_hw_workaround_match(hw, w)) { 12434 switch(w->workaround) { 12435 case HW_WORKAROUND_TEST: { 12436 ocs_log_debug(hw->os, "Override: test: %d\n", w->value); 12437 break; 12438 } 12439 12440 case HW_WORKAROUND_RETAIN_TSEND_IO_LENGTH: { 12441 ocs_log_debug(hw->os, "HW Workaround: retain TSEND IO length\n"); 12442 hw->workaround.retain_tsend_io_length = 1; 12443 break; 12444 } 12445 case HW_WORKAROUND_MAX_QUEUE: { 12446 sli4_qtype_e q; 12447 12448 ocs_log_debug(hw->os, "HW Workaround: override max_qentries: %d\n", w->value); 12449 for (q = SLI_QTYPE_EQ; q < SLI_QTYPE_MAX; q++) { 12450 if (hw->num_qentries[q] > w->value) { 12451 hw->num_qentries[q] = w->value; 12452 } 12453 } 12454 break; 12455 } 12456 case HW_WORKAROUND_MAX_RQ: { 12457 ocs_log_debug(hw->os, "HW Workaround: override RQ max_qentries: %d\n", w->value); 12458 if (hw->num_qentries[SLI_QTYPE_RQ] > w->value) { 12459 hw->num_qentries[SLI_QTYPE_RQ] = w->value; 12460 } 12461 break; 12462 } 12463 case HW_WORKAROUND_WQE_COUNT_METHOD: { 12464 ocs_log_debug(hw->os, "HW Workaround: set WQE count method=%d\n", w->value); 12465 sli4->config.count_method[SLI_QTYPE_WQ] = w->value; 12466 sli_calc_max_qentries(sli4); 12467 break; 12468 } 12469 case HW_WORKAROUND_RQE_COUNT_METHOD: { 12470 ocs_log_debug(hw->os, "HW Workaround: set RQE count method=%d\n", w->value); 12471 sli4->config.count_method[SLI_QTYPE_RQ] = w->value; 12472 sli_calc_max_qentries(sli4); 12473 break; 12474 } 12475 case HW_WORKAROUND_USE_UNREGISTERD_RPI: 12476 ocs_log_debug(hw->os, "HW Workaround: use unreg'd RPI if rnode->indicator == 0xFFFF\n"); 12477 hw->workaround.use_unregistered_rpi = TRUE; 12478 /* 12479 * Allocate an RPI that is never registered, to be used in the case where 12480 * a node has been unregistered, and its indicator (RPI) value is set to 0xFFFF 12481 */ 12482 if (sli_resource_alloc(&hw->sli, SLI_RSRC_FCOE_RPI, &hw->workaround.unregistered_rid, 12483 &hw->workaround.unregistered_index)) { 12484 ocs_log_err(hw->os, "sli_resource_alloc unregistered RPI failed\n"); 12485 hw->workaround.use_unregistered_rpi = FALSE; 12486 } 12487 break; 12488 case HW_WORKAROUND_DISABLE_AR_TGT_DIF: 12489 ocs_log_debug(hw->os, "HW Workaround: disable AR on T10-PI TSEND\n"); 12490 hw->workaround.disable_ar_tgt_dif = TRUE; 12491 break; 12492 case HW_WORKAROUND_DISABLE_SET_DUMP_LOC: 12493 ocs_log_debug(hw->os, "HW Workaround: disable set_dump_loc\n"); 12494 hw->workaround.disable_dump_loc = TRUE; 12495 break; 12496 case HW_WORKAROUND_USE_DIF_QUARANTINE: 12497 ocs_log_debug(hw->os, "HW Workaround: use DIF quarantine\n"); 12498 hw->workaround.use_dif_quarantine = TRUE; 12499 break; 12500 case HW_WORKAROUND_USE_DIF_SEC_XRI: 12501 ocs_log_debug(hw->os, "HW Workaround: use DIF secondary xri\n"); 12502 hw->workaround.use_dif_sec_xri = TRUE; 12503 break; 12504 case HW_WORKAROUND_OVERRIDE_FCFI_IN_SRB: 12505 ocs_log_debug(hw->os, "HW Workaround: override FCFI in SRB\n"); 12506 hw->workaround.override_fcfi = TRUE; 12507 break; 12508 12509 case HW_WORKAROUND_FW_VERSION_TOO_LOW: 12510 ocs_log_debug(hw->os, "HW Workaround: fw version is below the minimum for this driver\n"); 12511 hw->workaround.fw_version_too_low = TRUE; 12512 break; 12513 case HW_WORKAROUND_SGLC_MISREPORTED: 12514 ocs_log_debug(hw->os, "HW Workaround: SGLC misreported - chaining is enabled\n"); 12515 hw->workaround.sglc_misreported = TRUE; 12516 break; 12517 case HW_WORKAROUND_IGNORE_SEND_FRAME_CAPABLE: 12518 ocs_log_debug(hw->os, "HW Workaround: not SEND_FRAME capable - disabled\n"); 12519 hw->workaround.ignore_send_frame = TRUE; 12520 break; 12521 } /* switch(w->workaround) */ 12522 } 12523 } 12524 } 12525