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 * @defgroup sli SLI-4 Base APIs 36 */ 37 38 /** 39 * @file 40 * All common (i.e. transport-independent) SLI-4 functions are implemented 41 * in this file. 42 */ 43 44 #include "sli4.h" 45 46 #if defined(OCS_INCLUDE_DEBUG) 47 #include "ocs_utils.h" 48 #endif 49 50 #define SLI4_BMBX_DELAY_US 1000 /* 1 ms */ 51 #define SLI4_INIT_PORT_DELAY_US 10000 /* 10 ms */ 52 53 static int32_t sli_fw_init(sli4_t *); 54 static int32_t sli_fw_term(sli4_t *); 55 static int32_t sli_sliport_control(sli4_t *sli4, uint32_t endian); 56 static int32_t sli_cmd_fw_deinitialize(sli4_t *, void *, size_t); 57 static int32_t sli_cmd_fw_initialize(sli4_t *, void *, size_t); 58 static int32_t sli_queue_doorbell(sli4_t *, sli4_queue_t *); 59 static uint8_t sli_queue_entry_is_valid(sli4_queue_t *, uint8_t *, uint8_t); 60 61 const uint8_t sli4_fw_initialize[] = { 62 0xff, 0x12, 0x34, 0xff, 63 0xff, 0x56, 0x78, 0xff, 64 }; 65 66 const uint8_t sli4_fw_deinitialize[] = { 67 0xff, 0xaa, 0xbb, 0xff, 68 0xff, 0xcc, 0xdd, 0xff, 69 }; 70 71 typedef struct { 72 uint32_t rev_id; 73 uint32_t family; /* generation */ 74 sli4_asic_type_e type; 75 sli4_asic_rev_e rev; 76 } sli4_asic_entry_t; 77 78 sli4_asic_entry_t sli4_asic_table[] = { 79 { 0x00, 1, SLI4_ASIC_TYPE_BE3, SLI4_ASIC_REV_A0}, 80 { 0x01, 1, SLI4_ASIC_TYPE_BE3, SLI4_ASIC_REV_A1}, 81 { 0x02, 1, SLI4_ASIC_TYPE_BE3, SLI4_ASIC_REV_A2}, 82 { 0x00, 4, SLI4_ASIC_TYPE_SKYHAWK, SLI4_ASIC_REV_A0}, 83 { 0x00, 2, SLI4_ASIC_TYPE_SKYHAWK, SLI4_ASIC_REV_A0}, 84 { 0x10, 1, SLI4_ASIC_TYPE_BE3, SLI4_ASIC_REV_B0}, 85 { 0x10, 0x04, SLI4_ASIC_TYPE_SKYHAWK, SLI4_ASIC_REV_B0}, 86 { 0x11, 0x04, SLI4_ASIC_TYPE_SKYHAWK, SLI4_ASIC_REV_B1}, 87 { 0x0, 0x0a, SLI4_ASIC_TYPE_LANCER, SLI4_ASIC_REV_A0}, 88 { 0x10, 0x0b, SLI4_ASIC_TYPE_LANCER, SLI4_ASIC_REV_B0}, 89 { 0x30, 0x0b, SLI4_ASIC_TYPE_LANCER, SLI4_ASIC_REV_D0}, 90 { 0x3, 0x0b, SLI4_ASIC_TYPE_LANCERG6,SLI4_ASIC_REV_A3}, 91 { 0x0, 0x0c, SLI4_ASIC_TYPE_LANCERG6,SLI4_ASIC_REV_A0}, 92 { 0x1, 0x0c, SLI4_ASIC_TYPE_LANCERG6,SLI4_ASIC_REV_A1}, 93 { 0x3, 0x0c, SLI4_ASIC_TYPE_LANCERG6,SLI4_ASIC_REV_A3}, 94 { 0x1, 0x0d, SLI4_ASIC_TYPE_LANCERG7,SLI4_ASIC_REV_A1}, 95 { 0x10, 0x0d, SLI4_ASIC_TYPE_LANCERG7,SLI4_ASIC_REV_B0}, 96 { 0x00, 0x05, SLI4_ASIC_TYPE_CORSAIR, SLI4_ASIC_REV_A0}, 97 }; 98 99 /* 100 * @brief Convert queue type enum (SLI_QTYPE_*) into a string. 101 */ 102 const char *SLI_QNAME[] = { 103 "Event Queue", 104 "Completion Queue", 105 "Mailbox Queue", 106 "Work Queue", 107 "Receive Queue", 108 "Undefined" 109 }; 110 111 /** 112 * @brief Define the mapping of registers to their BAR and offset. 113 * 114 * @par Description 115 * Although SLI-4 specification defines a common set of registers, their locations 116 * (both BAR and offset) depend on the interface type. This array maps a register 117 * enum to an array of BAR/offset pairs indexed by the interface type. For 118 * example, to access the bootstrap mailbox register on an interface type 0 119 * device, code can refer to the offset using regmap[SLI4_REG_BMBX][0].offset. 120 * 121 * @b Note: A value of UINT32_MAX for either the register set (rset) or offset (off) 122 * indicates an invalid mapping. 123 */ 124 const sli4_reg_t regmap[SLI4_REG_MAX][SLI4_MAX_IF_TYPES] = { 125 /* SLI4_REG_BMBX */ 126 { 127 { 2, SLI4_BMBX_REG }, { 0, SLI4_BMBX_REG }, { 0, SLI4_BMBX_REG }, { 0, SLI4_BMBX_REG }, 128 { UINT32_MAX, UINT32_MAX }, { UINT32_MAX, UINT32_MAX } , { 0, SLI4_BMBX_REG }, 129 }, 130 /* SLI4_REG_EQCQ_DOORBELL */ 131 { 132 { 2, SLI4_EQCQ_DOORBELL_REG }, { 0, SLI4_EQCQ_DOORBELL_REG }, 133 { 0, SLI4_EQCQ_DOORBELL_REG }, { 0, SLI4_EQCQ_DOORBELL_REG }, 134 { UINT32_MAX, UINT32_MAX }, { UINT32_MAX, UINT32_MAX }, 135 { 1, SLI4_IF6_EQ_DOORBELL_REG } 136 }, 137 // SLI4_REG_CQ_DOORBELL 138 { 139 { 2, SLI4_EQCQ_DOORBELL_REG }, { 0, SLI4_EQCQ_DOORBELL_REG }, 140 { 0, SLI4_EQCQ_DOORBELL_REG }, { 0, SLI4_EQCQ_DOORBELL_REG }, 141 { UINT32_MAX, UINT32_MAX }, { UINT32_MAX, UINT32_MAX }, 142 { 1, SLI4_IF6_CQ_DOORBELL_REG } 143 }, 144 /* SLI4_REG_FCOE_RQ_DOORBELL */ 145 { 146 { 2, SLI4_RQ_DOORBELL_REG }, { 0, SLI4_RQ_DOORBELL_REG }, 147 { 0, SLI4_RQ_DOORBELL_REG }, { UINT32_MAX, UINT32_MAX }, 148 { UINT32_MAX, UINT32_MAX }, { UINT32_MAX, UINT32_MAX }, 149 { 1, SLI4_IF6_RQ_DOORBELL_REG } 150 }, 151 /* SLI4_REG_IO_WQ_DOORBELL */ 152 { 153 { 2, SLI4_IO_WQ_DOORBELL_REG }, { 0, SLI4_IO_WQ_DOORBELL_REG }, 154 { 0, SLI4_IO_WQ_DOORBELL_REG }, { UINT32_MAX, UINT32_MAX }, 155 { UINT32_MAX, UINT32_MAX }, { UINT32_MAX, UINT32_MAX }, 156 { 1, SLI4_IF6_WQ_DOORBELL_REG } 157 }, 158 /* SLI4_REG_MQ_DOORBELL */ 159 { 160 { 2, SLI4_MQ_DOORBELL_REG }, { 0, SLI4_MQ_DOORBELL_REG }, 161 { 0, SLI4_MQ_DOORBELL_REG }, { 0, SLI4_MQ_DOORBELL_REG }, 162 { UINT32_MAX, UINT32_MAX }, { UINT32_MAX, UINT32_MAX }, 163 { 1, SLI4_IF6_MQ_DOORBELL_REG } 164 }, 165 /* SLI4_REG_PHYSDEV_CONTROL */ 166 { 167 { UINT32_MAX, UINT32_MAX }, { UINT32_MAX, UINT32_MAX }, 168 { 0, SLI4_PHSDEV_CONTROL_REG_236 }, { 0, SLI4_PHSDEV_CONTROL_REG_236 }, 169 { UINT32_MAX, UINT32_MAX }, { UINT32_MAX, UINT32_MAX }, 170 { 0, SLI4_PHSDEV_CONTROL_REG_236 } 171 }, 172 /* SLI4_REG_SLIPORT_CONTROL */ 173 { 174 { UINT32_MAX, UINT32_MAX }, { UINT32_MAX, UINT32_MAX }, 175 { 0, SLI4_SLIPORT_CONTROL_REG }, { UINT32_MAX, UINT32_MAX }, 176 { UINT32_MAX, UINT32_MAX }, { UINT32_MAX, UINT32_MAX }, 177 { 0, SLI4_SLIPORT_CONTROL_REG }, 178 }, 179 /* SLI4_REG_SLIPORT_ERROR1 */ 180 { 181 { UINT32_MAX, UINT32_MAX }, { UINT32_MAX, UINT32_MAX }, 182 { 0, SLI4_SLIPORT_ERROR1 }, { UINT32_MAX, UINT32_MAX }, 183 { UINT32_MAX, UINT32_MAX }, { UINT32_MAX, UINT32_MAX }, 184 { 0, SLI4_SLIPORT_ERROR1 }, 185 }, 186 /* SLI4_REG_SLIPORT_ERROR2 */ 187 { 188 { UINT32_MAX, UINT32_MAX }, { UINT32_MAX, UINT32_MAX }, 189 { 0, SLI4_SLIPORT_ERROR2 }, { UINT32_MAX, UINT32_MAX }, 190 { UINT32_MAX, UINT32_MAX }, { UINT32_MAX, UINT32_MAX }, 191 { 0, SLI4_SLIPORT_ERROR2 }, 192 }, 193 /* SLI4_REG_SLIPORT_SEMAPHORE */ 194 { 195 { 1, SLI4_PORT_SEMAPHORE_REG_0 }, { 0, SLI4_PORT_SEMAPHORE_REG_1 }, 196 { 0, SLI4_PORT_SEMAPHORE_REG_236 }, { 0, SLI4_PORT_SEMAPHORE_REG_236 }, 197 { UINT32_MAX, UINT32_MAX }, { UINT32_MAX, UINT32_MAX }, 198 { 0, SLI4_PORT_SEMAPHORE_REG_236 }, 199 }, 200 /* SLI4_REG_SLIPORT_STATUS */ 201 { 202 { UINT32_MAX, UINT32_MAX }, { UINT32_MAX, UINT32_MAX }, 203 { 0, SLI4_PORT_STATUS_REG_236 }, { 0, SLI4_PORT_STATUS_REG_236 }, 204 { UINT32_MAX, UINT32_MAX }, { UINT32_MAX, UINT32_MAX }, 205 { 0, SLI4_PORT_STATUS_REG_236 }, 206 }, 207 /* SLI4_REG_UERR_MASK_HI */ 208 { 209 { 0, SLI4_UERR_MASK_HIGH_REG }, { UINT32_MAX, UINT32_MAX }, 210 { UINT32_MAX, UINT32_MAX }, { UINT32_MAX, UINT32_MAX }, 211 { UINT32_MAX, UINT32_MAX }, { UINT32_MAX, UINT32_MAX }, 212 { UINT32_MAX, UINT32_MAX } 213 }, 214 /* SLI4_REG_UERR_MASK_LO */ 215 { 216 { 0, SLI4_UERR_MASK_LOW_REG }, { UINT32_MAX, UINT32_MAX }, 217 { UINT32_MAX, UINT32_MAX }, { UINT32_MAX, UINT32_MAX }, 218 { UINT32_MAX, UINT32_MAX }, { UINT32_MAX, UINT32_MAX }, 219 { UINT32_MAX, UINT32_MAX } 220 }, 221 /* SLI4_REG_UERR_STATUS_HI */ 222 { 223 { 0, SLI4_UERR_STATUS_HIGH_REG }, { UINT32_MAX, UINT32_MAX }, 224 { UINT32_MAX, UINT32_MAX }, { UINT32_MAX, UINT32_MAX }, 225 { UINT32_MAX, UINT32_MAX }, { UINT32_MAX, UINT32_MAX }, 226 { UINT32_MAX, UINT32_MAX } 227 }, 228 /* SLI4_REG_UERR_STATUS_LO */ 229 { 230 { 0, SLI4_UERR_STATUS_LOW_REG }, { UINT32_MAX, UINT32_MAX }, 231 { UINT32_MAX, UINT32_MAX }, { UINT32_MAX, UINT32_MAX }, 232 { UINT32_MAX, UINT32_MAX }, { UINT32_MAX, UINT32_MAX }, 233 { UINT32_MAX, UINT32_MAX } 234 }, 235 /* SLI4_REG_SW_UE_CSR1 */ 236 { 237 { 1, SLI4_SW_UE_CSR1}, { UINT32_MAX, UINT32_MAX }, 238 { UINT32_MAX, UINT32_MAX }, { UINT32_MAX, UINT32_MAX }, 239 { UINT32_MAX, UINT32_MAX }, { UINT32_MAX, UINT32_MAX }, 240 { UINT32_MAX, UINT32_MAX } 241 }, 242 /* SLI4_REG_SW_UE_CSR2 */ 243 { 244 { 1, SLI4_SW_UE_CSR2}, { UINT32_MAX, UINT32_MAX }, 245 { UINT32_MAX, UINT32_MAX }, { UINT32_MAX, UINT32_MAX }, 246 { UINT32_MAX, UINT32_MAX }, { UINT32_MAX, UINT32_MAX }, 247 { UINT32_MAX, UINT32_MAX } 248 }, 249 }; 250 251 /** 252 * @brief Read the given SLI register. 253 * 254 * @param sli Pointer to the SLI context. 255 * @param reg Register name enum. 256 * 257 * @return Returns the register value. 258 */ 259 uint32_t 260 sli_reg_read(sli4_t *sli, sli4_regname_e reg) 261 { 262 const sli4_reg_t *r = &(regmap[reg][sli->if_type]); 263 264 if ((UINT32_MAX == r->rset) || (UINT32_MAX == r->off)) { 265 ocs_log_err(sli->os, "regname %d not defined for if_type %d\n", reg, sli->if_type); 266 return UINT32_MAX; 267 } 268 269 return ocs_reg_read32(sli->os, r->rset, r->off); 270 } 271 272 /** 273 * @brief Write the value to the given SLI register. 274 * 275 * @param sli Pointer to the SLI context. 276 * @param reg Register name enum. 277 * @param val Value to write. 278 * 279 * @return None. 280 */ 281 void 282 sli_reg_write(sli4_t *sli, sli4_regname_e reg, uint32_t val) 283 { 284 const sli4_reg_t *r = &(regmap[reg][sli->if_type]); 285 286 if ((UINT32_MAX == r->rset) || (UINT32_MAX == r->off)) { 287 ocs_log_err(sli->os, "regname %d not defined for if_type %d\n", reg, sli->if_type); 288 return; 289 } 290 291 ocs_reg_write32(sli->os, r->rset, r->off, val); 292 } 293 294 /** 295 * @brief Check if the SLI_INTF register is valid. 296 * 297 * @param val 32-bit SLI_INTF register value. 298 * 299 * @return Returns 0 on success, or a non-zero value on failure. 300 */ 301 static uint8_t 302 sli_intf_valid_check(uint32_t val) 303 { 304 return ((val >> SLI4_INTF_VALID_SHIFT) & SLI4_INTF_VALID_MASK) != SLI4_INTF_VALID; 305 } 306 307 /** 308 * @brief Retrieve the SLI revision level. 309 * 310 * @param val 32-bit SLI_INTF register value. 311 * 312 * @return Returns the SLI revision level. 313 */ 314 static uint8_t 315 sli_intf_sli_revision(uint32_t val) 316 { 317 return ((val >> SLI4_INTF_SLI_REVISION_SHIFT) & SLI4_INTF_SLI_REVISION_MASK); 318 } 319 320 static uint8_t 321 sli_intf_sli_family(uint32_t val) 322 { 323 return ((val >> SLI4_INTF_SLI_FAMILY_SHIFT) & SLI4_INTF_SLI_FAMILY_MASK); 324 } 325 326 /** 327 * @brief Retrieve the SLI interface type. 328 * 329 * @param val 32-bit SLI_INTF register value. 330 * 331 * @return Returns the SLI interface type. 332 */ 333 static uint8_t 334 sli_intf_if_type(uint32_t val) 335 { 336 return ((val >> SLI4_INTF_IF_TYPE_SHIFT) & SLI4_INTF_IF_TYPE_MASK); 337 } 338 339 /** 340 * @brief Retrieve PCI revision ID. 341 * 342 * @param val 32-bit PCI CLASS_REVISION register value. 343 * 344 * @return Returns the PCI revision ID. 345 */ 346 static uint8_t 347 sli_pci_rev_id(uint32_t val) 348 { 349 return ((val >> SLI4_PCI_REV_ID_SHIFT) & SLI4_PCI_REV_ID_MASK); 350 } 351 352 /** 353 * @brief retrieve SLI ASIC generation 354 * 355 * @param val 32-bit SLI_ASIC_ID register value 356 * 357 * @return SLI ASIC generation 358 */ 359 static uint8_t 360 sli_asic_gen(uint32_t val) 361 { 362 return ((val >> SLI4_ASIC_GEN_SHIFT) & SLI4_ASIC_GEN_MASK); 363 } 364 365 /** 366 * @brief Wait for the bootstrap mailbox to report "ready". 367 * 368 * @param sli4 SLI context pointer. 369 * @param msec Number of milliseconds to wait. 370 * 371 * @return Returns 0 if BMBX is ready, or non-zero otherwise (i.e. time out occurred). 372 */ 373 static int32_t 374 sli_bmbx_wait(sli4_t *sli4, uint32_t msec) 375 { 376 uint32_t val = 0; 377 378 do { 379 ocs_udelay(SLI4_BMBX_DELAY_US); 380 val = sli_reg_read(sli4, SLI4_REG_BMBX); 381 msec--; 382 } while(msec && !(val & SLI4_BMBX_RDY)); 383 384 return(!(val & SLI4_BMBX_RDY)); 385 } 386 387 /** 388 * @brief Write bootstrap mailbox. 389 * 390 * @param sli4 SLI context pointer. 391 * 392 * @return Returns 0 if command succeeded, or non-zero otherwise. 393 */ 394 static int32_t 395 sli_bmbx_write(sli4_t *sli4) 396 { 397 uint32_t val = 0; 398 399 /* write buffer location to bootstrap mailbox register */ 400 ocs_dma_sync(&sli4->bmbx, OCS_DMASYNC_PREWRITE); 401 val = SLI4_BMBX_WRITE_HI(sli4->bmbx.phys); 402 sli_reg_write(sli4, SLI4_REG_BMBX, val); 403 404 if (sli_bmbx_wait(sli4, SLI4_BMBX_DELAY_US)) { 405 ocs_log_crit(sli4->os, "BMBX WRITE_HI failed\n"); 406 return -1; 407 } 408 val = SLI4_BMBX_WRITE_LO(sli4->bmbx.phys); 409 sli_reg_write(sli4, SLI4_REG_BMBX, val); 410 411 /* wait for SLI Port to set ready bit */ 412 return sli_bmbx_wait(sli4, SLI4_BMBX_TIMEOUT_MSEC/*XXX*/); 413 } 414 415 #if defined(OCS_INCLUDE_DEBUG) 416 /** 417 * @ingroup sli 418 * @brief Dump BMBX mailbox command. 419 * 420 * @par Description 421 * Convenience function for dumping BMBX mailbox commands. Takes 422 * into account which mailbox command is given since SLI_CONFIG 423 * commands are special. 424 * 425 * @b Note: This function takes advantage of 426 * the one-command-at-a-time nature of the BMBX to be able to 427 * display non-embedded SLI_CONFIG commands. This will not work 428 * for mailbox commands on the MQ. Luckily, all current non-emb 429 * mailbox commands go through the BMBX. 430 * 431 * @param sli4 SLI context pointer. 432 * @param mbx Pointer to mailbox command to dump. 433 * @param prefix Prefix for dump label. 434 * 435 * @return None. 436 */ 437 static void 438 sli_dump_bmbx_command(sli4_t *sli4, void *mbx, const char *prefix) 439 { 440 uint32_t size = 0; 441 char label[64]; 442 uint32_t i; 443 /* Mailbox diagnostic logging */ 444 sli4_mbox_command_header_t *hdr = (sli4_mbox_command_header_t *)mbx; 445 446 if (!ocs_debug_is_enabled(OCS_DEBUG_ENABLE_MQ_DUMP)) { 447 return; 448 } 449 450 if (hdr->command == SLI4_MBOX_COMMAND_SLI_CONFIG) { 451 sli4_cmd_sli_config_t *sli_config = (sli4_cmd_sli_config_t *)hdr; 452 sli4_req_hdr_t *sli_config_hdr; 453 if (sli_config->emb) { 454 ocs_snprintf(label, sizeof(label), "%s (emb)", prefix); 455 456 /* if embedded, dump entire command */ 457 sli_config_hdr = (sli4_req_hdr_t *)sli_config->payload.embed; 458 size = sizeof(*sli_config) - sizeof(sli_config->payload) + 459 sli_config_hdr->request_length + (4*sizeof(uint32_t)); 460 ocs_dump32(OCS_DEBUG_ENABLE_MQ_DUMP, sli4->os, label, 461 (uint8_t *)sli4->bmbx.virt, size); 462 } else { 463 sli4_sli_config_pmd_t *pmd; 464 ocs_snprintf(label, sizeof(label), "%s (non-emb hdr)", prefix); 465 466 /* if non-embedded, break up into two parts: SLI_CONFIG hdr 467 and the payload(s) */ 468 size = sizeof(*sli_config) - sizeof(sli_config->payload) + (12 * sli_config->pmd_count); 469 ocs_dump32(OCS_DEBUG_ENABLE_MQ_DUMP, sli4->os, label, 470 (uint8_t *)sli4->bmbx.virt, size); 471 472 /* as sanity check, make sure first PMD matches what was saved */ 473 pmd = &sli_config->payload.mem; 474 if ((pmd->address_high == ocs_addr32_hi(sli4->bmbx_non_emb_pmd->phys)) && 475 (pmd->address_low == ocs_addr32_lo(sli4->bmbx_non_emb_pmd->phys))) { 476 for (i = 0; i < sli_config->pmd_count; i++, pmd++) { 477 sli_config_hdr = sli4->bmbx_non_emb_pmd->virt; 478 ocs_snprintf(label, sizeof(label), "%s (non-emb pay[%d])", 479 prefix, i); 480 ocs_dump32(OCS_DEBUG_ENABLE_MQ_DUMP, sli4->os, label, 481 (uint8_t *)sli4->bmbx_non_emb_pmd->virt, 482 sli_config_hdr->request_length + (4*sizeof(uint32_t))); 483 } 484 } else { 485 ocs_log_debug(sli4->os, "pmd addr does not match pmd:%x %x (%x %x)\n", 486 pmd->address_high, pmd->address_low, 487 ocs_addr32_hi(sli4->bmbx_non_emb_pmd->phys), 488 ocs_addr32_lo(sli4->bmbx_non_emb_pmd->phys)); 489 } 490 } 491 } else { 492 /* not an SLI_CONFIG command, just display first 64 bytes, like we do 493 for MQEs */ 494 size = 64; 495 ocs_dump32(OCS_DEBUG_ENABLE_MQ_DUMP, sli4->os, prefix, 496 (uint8_t *)mbx, size); 497 } 498 } 499 #endif 500 501 /** 502 * @ingroup sli 503 * @brief Submit a command to the bootstrap mailbox and check the status. 504 * 505 * @param sli4 SLI context pointer. 506 * 507 * @return Returns 0 on success, or a non-zero value on failure. 508 */ 509 int32_t 510 sli_bmbx_command(sli4_t *sli4) 511 { 512 void *cqe = (uint8_t *)sli4->bmbx.virt + SLI4_BMBX_SIZE; 513 514 #if defined(OCS_INCLUDE_DEBUG) 515 sli_dump_bmbx_command(sli4, sli4->bmbx.virt, "bmbx cmd"); 516 #endif 517 518 if (sli_fw_error_status(sli4) > 0) { 519 ocs_log_crit(sli4->os, "Chip is in an error state - Mailbox " 520 "command rejected status=%#x error1=%#x error2=%#x\n", 521 sli_reg_read(sli4, SLI4_REG_SLIPORT_STATUS), 522 sli_reg_read(sli4, SLI4_REG_SLIPORT_ERROR1), 523 sli_reg_read(sli4, SLI4_REG_SLIPORT_ERROR2)); 524 return -1; 525 } 526 527 if (sli_bmbx_write(sli4)) { 528 ocs_log_crit(sli4->os, "bootstrap mailbox write fail phys=%p reg=%#x\n", 529 (void*)sli4->bmbx.phys, 530 sli_reg_read(sli4, SLI4_REG_BMBX)); 531 return -1; 532 } 533 534 /* check completion queue entry status */ 535 ocs_dma_sync(&sli4->bmbx, OCS_DMASYNC_POSTREAD); 536 if (((sli4_mcqe_t *)cqe)->val) { 537 #if defined(OCS_INCLUDE_DEBUG) 538 sli_dump_bmbx_command(sli4, sli4->bmbx.virt, "bmbx cmpl"); 539 ocs_dump32(OCS_DEBUG_ENABLE_CQ_DUMP, sli4->os, "bmbx cqe", cqe, sizeof(sli4_mcqe_t)); 540 #endif 541 return sli_cqe_mq(cqe); 542 } else { 543 ocs_log_err(sli4->os, "invalid or wrong type\n"); 544 return -1; 545 } 546 } 547 548 /**************************************************************************** 549 * Messages 550 */ 551 552 /** 553 * @ingroup sli 554 * @brief Write a CONFIG_LINK command to the provided buffer. 555 * 556 * @param sli4 SLI context pointer. 557 * @param buf Virtual pointer to the destination buffer. 558 * @param size Buffer size, in bytes. 559 * 560 * @return Returns the number of bytes written. 561 */ 562 int32_t 563 sli_cmd_config_link(sli4_t *sli4, void *buf, size_t size) 564 { 565 sli4_cmd_config_link_t *config_link = buf; 566 567 ocs_memset(buf, 0, size); 568 569 config_link->hdr.command = SLI4_MBOX_COMMAND_CONFIG_LINK; 570 571 /* Port interprets zero in a field as "use default value" */ 572 573 return sizeof(sli4_cmd_config_link_t); 574 } 575 576 /** 577 * @ingroup sli 578 * @brief Write a DOWN_LINK command to the provided buffer. 579 * 580 * @param sli4 SLI context pointer. 581 * @param buf Virtual pointer to the destination buffer. 582 * @param size Buffer size, in bytes. 583 * 584 * @return Returns the number of bytes written. 585 */ 586 int32_t 587 sli_cmd_down_link(sli4_t *sli4, void *buf, size_t size) 588 { 589 sli4_mbox_command_header_t *hdr = buf; 590 591 ocs_memset(buf, 0, size); 592 593 hdr->command = SLI4_MBOX_COMMAND_DOWN_LINK; 594 595 /* Port interprets zero in a field as "use default value" */ 596 597 return sizeof(sli4_mbox_command_header_t); 598 } 599 600 /** 601 * @ingroup sli 602 * @brief Write a DUMP Type 4 command to the provided buffer. 603 * 604 * @param sli4 SLI context pointer. 605 * @param buf Virtual pointer to the destination buffer. 606 * @param size Buffer size, in bytes. 607 * @param wki The well known item ID. 608 * 609 * @return Returns the number of bytes written. 610 */ 611 int32_t 612 sli_cmd_dump_type4(sli4_t *sli4, void *buf, size_t size, uint16_t wki) 613 { 614 sli4_cmd_dump4_t *cmd = buf; 615 616 ocs_memset(buf, 0, size); 617 618 cmd->hdr.command = SLI4_MBOX_COMMAND_DUMP; 619 cmd->type = 4; 620 cmd->wki_selection = wki; 621 return sizeof(sli4_cmd_dump4_t); 622 } 623 624 /** 625 * @ingroup sli 626 * @brief Write a COMMON_READ_TRANSCEIVER_DATA command. 627 * 628 * @param sli4 SLI context. 629 * @param buf Destination buffer for the command. 630 * @param size Buffer size, in bytes. 631 * @param page_num The page of SFP data to retrieve (0xa0 or 0xa2). 632 * @param dma DMA structure from which the data will be copied. 633 * 634 * @note This creates a Version 0 message. 635 * 636 * @return Returns the number of bytes written. 637 */ 638 int32_t 639 sli_cmd_common_read_transceiver_data(sli4_t *sli4, void *buf, size_t size, uint32_t page_num, 640 ocs_dma_t *dma) 641 { 642 sli4_req_common_read_transceiver_data_t *req = NULL; 643 uint32_t sli_config_off = 0; 644 uint32_t payload_size; 645 646 if (dma == NULL) { 647 /* Payload length must accommodate both request and response */ 648 payload_size = max(sizeof(sli4_req_common_read_transceiver_data_t), 649 sizeof(sli4_res_common_read_transceiver_data_t)); 650 } else { 651 payload_size = dma->size; 652 } 653 654 if (sli4->port_type == SLI4_PORT_TYPE_FC) { 655 sli_config_off = sli_cmd_sli_config(sli4, buf, size, payload_size, dma); 656 } 657 658 if (dma == NULL) { 659 req = (sli4_req_common_read_transceiver_data_t *)((uint8_t *)buf + sli_config_off); 660 } else { 661 req = (sli4_req_common_read_transceiver_data_t *)dma->virt; 662 ocs_memset(req, 0, dma->size); 663 } 664 665 req->hdr.opcode = SLI4_OPC_COMMON_READ_TRANSCEIVER_DATA; 666 req->hdr.subsystem = SLI4_SUBSYSTEM_COMMON; 667 req->hdr.request_length = payload_size - sizeof(sli4_req_hdr_t); 668 669 req->page_number = page_num; 670 req->port = sli4->physical_port; 671 672 return(sli_config_off + sizeof(sli4_req_common_read_transceiver_data_t)); 673 } 674 675 /** 676 * @ingroup sli 677 * @brief Write a READ_LINK_STAT command to the provided buffer. 678 * 679 * @param sli4 SLI context pointer. 680 * @param buf Virtual pointer to the destination buffer. 681 * @param size Buffer size, in bytes. 682 * @param req_ext_counters If TRUE, then the extended counters will be requested. 683 * @param clear_overflow_flags If TRUE, then overflow flags will be cleared. 684 * @param clear_all_counters If TRUE, the counters will be cleared. 685 * 686 * @return Returns the number of bytes written. 687 */ 688 int32_t 689 sli_cmd_read_link_stats(sli4_t *sli4, void *buf, size_t size, 690 uint8_t req_ext_counters, 691 uint8_t clear_overflow_flags, 692 uint8_t clear_all_counters) 693 { 694 sli4_cmd_read_link_stats_t *cmd = buf; 695 696 ocs_memset(buf, 0, size); 697 698 cmd->hdr.command = SLI4_MBOX_COMMAND_READ_LNK_STAT; 699 cmd->rec = req_ext_counters; 700 cmd->clrc = clear_all_counters; 701 cmd->clof = clear_overflow_flags; 702 return sizeof(sli4_cmd_read_link_stats_t); 703 } 704 705 /** 706 * @ingroup sli 707 * @brief Write a READ_STATUS command to the provided buffer. 708 * 709 * @param sli4 SLI context pointer. 710 * @param buf Virtual pointer to the destination buffer. 711 * @param size Buffer size, in bytes. 712 * @param clear_counters If TRUE, the counters will be cleared. 713 * 714 * @return Returns the number of bytes written. 715 */ 716 int32_t 717 sli_cmd_read_status(sli4_t *sli4, void *buf, size_t size, 718 uint8_t clear_counters) 719 { 720 sli4_cmd_read_status_t *cmd = buf; 721 722 ocs_memset(buf, 0, size); 723 724 cmd->hdr.command = SLI4_MBOX_COMMAND_READ_STATUS; 725 cmd->cc = clear_counters; 726 return sizeof(sli4_cmd_read_status_t); 727 } 728 729 /** 730 * @brief Write a FW_DEINITIALIZE command to the provided buffer. 731 * 732 * @param sli4 SLI context pointer. 733 * @param buf Virtual pointer to the destination buffer. 734 * @param size Buffer size, in bytes. 735 * 736 * @return Returns the number of bytes written. 737 */ 738 static int32_t 739 sli_cmd_fw_deinitialize(sli4_t *sli4, void *buf, size_t size) 740 { 741 742 ocs_memset(buf, 0, size); 743 ocs_memcpy(buf, sli4_fw_deinitialize, sizeof(sli4_fw_deinitialize)); 744 745 return sizeof(sli4_fw_deinitialize); 746 } 747 748 /** 749 * @brief Write a FW_INITIALIZE command to the provided buffer. 750 * 751 * @param sli4 SLI context pointer. 752 * @param buf Virtual pointer to the destination buffer. 753 * @param size Buffer size, in bytes. 754 * 755 * @return Returns the number of bytes written. 756 */ 757 static int32_t 758 sli_cmd_fw_initialize(sli4_t *sli4, void *buf, size_t size) 759 { 760 761 ocs_memset(buf, 0, size); 762 ocs_memcpy(buf, sli4_fw_initialize, sizeof(sli4_fw_initialize)); 763 764 return sizeof(sli4_fw_initialize); 765 } 766 767 /** 768 * @ingroup sli 769 * @brief update INIT_LINK flags with the sli config topology. 770 * 771 * @param sli4 SLI context pointer. 772 * @param init_link Pointer to the init link command 773 * 774 * @return Returns 0 on success, -1 on failure 775 */ 776 static int32_t 777 sli4_set_link_flags_config_topo(sli4_t *sli4, sli4_cmd_init_link_t *init_link) 778 { 779 780 switch (sli4->config.topology) { 781 case SLI4_READ_CFG_TOPO_FC: 782 // Attempt P2P but failover to FC-AL 783 init_link->link_flags.enable_topology_failover = TRUE; 784 init_link->link_flags.topology = SLI4_INIT_LINK_F_P2P_FAIL_OVER; 785 break; 786 case SLI4_READ_CFG_TOPO_FC_AL: 787 init_link->link_flags.topology = SLI4_INIT_LINK_F_FCAL_ONLY; 788 return (!sli_fcal_is_speed_supported(init_link->link_speed_selection_code)); 789 790 case SLI4_READ_CFG_TOPO_FC_DA: 791 init_link->link_flags.topology = FC_TOPOLOGY_P2P; 792 break; 793 default: 794 ocs_log_err(sli4->os, "unsupported topology %#x\n", sli4->config.topology); 795 return -1; 796 } 797 798 return 0; 799 } 800 801 /** 802 * @ingroup sli 803 * @brief update INIT_LINK flags with the persistent topology. 804 * PT stores value in compatible form, directly assign to link_flags 805 * 806 * @param sli4 SLI context pointer. 807 * @param init_link Pointer to the init link command 808 * 809 * @return Returns 0 on success, -1 on failure 810 */ 811 static int32_t 812 sli4_set_link_flags_persistent_topo(sli4_t *sli4, sli4_cmd_init_link_t *init_link) 813 { 814 if ((sli4->config.pt == SLI4_INIT_LINK_F_FCAL_ONLY) && 815 (!sli_fcal_is_speed_supported(init_link->link_speed_selection_code))) 816 return -1; 817 818 init_link->link_flags.enable_topology_failover = sli4->config.tf; 819 init_link->link_flags.topology = sli4->config.pt; 820 821 return 0; 822 } 823 824 /** 825 * @ingroup sli 826 * @brief Write an INIT_LINK command to the provided buffer. 827 * 828 * @param sli4 SLI context pointer. 829 * @param buf Virtual pointer to the destination buffer. 830 * @param size Buffer size, in bytes. 831 * @param speed Link speed. 832 * @param reset_alpa For native FC, this is the selective reset AL_PA 833 * 834 * @return Returns the number of bytes written. 835 */ 836 int32_t 837 sli_cmd_init_link(sli4_t *sli4, void *buf, size_t size, uint32_t speed, uint8_t reset_alpa) 838 { 839 sli4_cmd_init_link_t *init_link = buf; 840 int32_t rc = 0; 841 842 ocs_memset(buf, 0, size); 843 844 init_link->hdr.command = SLI4_MBOX_COMMAND_INIT_LINK; 845 846 /* Most fields only have meaning for FC links */ 847 if (sli4->config.topology != SLI4_READ_CFG_TOPO_FCOE) { 848 init_link->selective_reset_al_pa = reset_alpa; 849 init_link->link_flags.loopback = FALSE; 850 851 init_link->link_speed_selection_code = speed; 852 switch (speed) { 853 case FC_LINK_SPEED_1G: 854 case FC_LINK_SPEED_2G: 855 case FC_LINK_SPEED_4G: 856 case FC_LINK_SPEED_8G: 857 case FC_LINK_SPEED_16G: 858 case FC_LINK_SPEED_32G: 859 init_link->link_flags.fixed_speed = TRUE; 860 break; 861 case FC_LINK_SPEED_10G: 862 ocs_log_test(sli4->os, "unsupported FC speed %d\n", speed); 863 return 0; 864 } 865 866 init_link->link_flags.unfair = FALSE; 867 init_link->link_flags.skip_lirp_lilp = FALSE; 868 init_link->link_flags.gen_loop_validity_check = FALSE; 869 init_link->link_flags.skip_lisa = FALSE; 870 init_link->link_flags.select_hightest_al_pa = FALSE; 871 872 //update topology in the link flags for link bring up 873 ocs_log_info(sli4->os, "bring up link with topology: %d, PTV: %d, TF: %d, PT: %d \n", 874 sli4->config.topology, sli4->config.ptv, sli4->config.tf, sli4->config.pt); 875 if (sli4->config.ptv) 876 rc = sli4_set_link_flags_persistent_topo(sli4, init_link); 877 else 878 rc = sli4_set_link_flags_config_topo(sli4, init_link); 879 880 } 881 882 return rc ? 0 : sizeof(sli4_cmd_init_link_t); 883 } 884 885 /** 886 * @ingroup sli 887 * @brief Write an INIT_VFI command to the provided buffer. 888 * 889 * @param sli4 SLI context pointer. 890 * @param buf Virtual pointer to the destination buffer. 891 * @param size Buffer size, in bytes. 892 * @param vfi VFI 893 * @param fcfi FCFI 894 * @param vpi VPI (Set to -1 if unused.) 895 * 896 * @return Returns the number of bytes written. 897 */ 898 int32_t 899 sli_cmd_init_vfi(sli4_t *sli4, void *buf, size_t size, uint16_t vfi, 900 uint16_t fcfi, uint16_t vpi) 901 { 902 sli4_cmd_init_vfi_t *init_vfi = buf; 903 904 ocs_memset(buf, 0, size); 905 906 init_vfi->hdr.command = SLI4_MBOX_COMMAND_INIT_VFI; 907 908 init_vfi->vfi = vfi; 909 init_vfi->fcfi = fcfi; 910 911 /* 912 * If the VPI is valid, initialize it at the same time as 913 * the VFI 914 */ 915 if (0xffff != vpi) { 916 init_vfi->vp = TRUE; 917 init_vfi->vpi = vpi; 918 } 919 920 return sizeof(sli4_cmd_init_vfi_t); 921 } 922 923 /** 924 * @ingroup sli 925 * @brief Write an INIT_VPI command to the provided buffer. 926 * 927 * @param sli4 SLI context pointer. 928 * @param buf Virtual pointer to the destination buffer. 929 * @param size Buffer size, in bytes. 930 * @param vpi VPI allocated. 931 * @param vfi VFI associated with this VPI. 932 * 933 * @return Returns the number of bytes written. 934 */ 935 int32_t 936 sli_cmd_init_vpi(sli4_t *sli4, void *buf, size_t size, uint16_t vpi, uint16_t vfi) 937 { 938 sli4_cmd_init_vpi_t *init_vpi = buf; 939 940 ocs_memset(buf, 0, size); 941 942 init_vpi->hdr.command = SLI4_MBOX_COMMAND_INIT_VPI; 943 init_vpi->vpi = vpi; 944 init_vpi->vfi = vfi; 945 946 return sizeof(sli4_cmd_init_vpi_t); 947 } 948 949 /** 950 * @ingroup sli 951 * @brief Write a POST_XRI command to the provided buffer. 952 * 953 * @param sli4 SLI context pointer. 954 * @param buf Virtual pointer to the destination buffer. 955 * @param size Buffer size, in bytes. 956 * @param xri_base Starting XRI value for range of XRI given to SLI Port. 957 * @param xri_count Number of XRIs provided to the SLI Port. 958 * 959 * @return Returns the number of bytes written. 960 */ 961 int32_t 962 sli_cmd_post_xri(sli4_t *sli4, void *buf, size_t size, uint16_t xri_base, uint16_t xri_count) 963 { 964 sli4_cmd_post_xri_t *post_xri = buf; 965 966 ocs_memset(buf, 0, size); 967 968 post_xri->hdr.command = SLI4_MBOX_COMMAND_POST_XRI; 969 post_xri->xri_base = xri_base; 970 post_xri->xri_count = xri_count; 971 972 if (sli4->config.auto_xfer_rdy == 0) { 973 post_xri->enx = TRUE; 974 post_xri->val = TRUE; 975 } 976 977 return sizeof(sli4_cmd_post_xri_t); 978 } 979 980 /** 981 * @ingroup sli 982 * @brief Write a RELEASE_XRI command to the provided buffer. 983 * 984 * @param sli4 SLI context pointer. 985 * @param buf Virtual pointer to the destination buffer. 986 * @param size Buffer size, in bytes. 987 * @param num_xri The number of XRIs to be released. 988 * 989 * @return Returns the number of bytes written. 990 */ 991 int32_t 992 sli_cmd_release_xri(sli4_t *sli4, void *buf, size_t size, uint8_t num_xri) 993 { 994 sli4_cmd_release_xri_t *release_xri = buf; 995 996 ocs_memset(buf, 0, size); 997 998 release_xri->hdr.command = SLI4_MBOX_COMMAND_RELEASE_XRI; 999 release_xri->xri_count = num_xri; 1000 1001 return sizeof(sli4_cmd_release_xri_t); 1002 } 1003 1004 /** 1005 * @brief Write a READ_CONFIG command to the provided buffer. 1006 * 1007 * @param sli4 SLI context pointer. 1008 * @param buf Virtual pointer to the destination buffer. 1009 * @param size Buffer size, in bytes 1010 * 1011 * @return Returns the number of bytes written. 1012 */ 1013 static int32_t 1014 sli_cmd_read_config(sli4_t *sli4, void *buf, size_t size) 1015 { 1016 sli4_cmd_read_config_t *read_config = buf; 1017 1018 ocs_memset(buf, 0, size); 1019 1020 read_config->hdr.command = SLI4_MBOX_COMMAND_READ_CONFIG; 1021 1022 return sizeof(sli4_cmd_read_config_t); 1023 } 1024 1025 /** 1026 * @brief Write a READ_NVPARMS command to the provided buffer. 1027 * 1028 * @param sli4 SLI context pointer. 1029 * @param buf Virtual pointer to the destination buffer. 1030 * @param size Buffer size, in bytes. 1031 * 1032 * @return Returns the number of bytes written. 1033 */ 1034 int32_t 1035 sli_cmd_read_nvparms(sli4_t *sli4, void *buf, size_t size) 1036 { 1037 sli4_cmd_read_nvparms_t *read_nvparms = buf; 1038 1039 ocs_memset(buf, 0, size); 1040 1041 read_nvparms->hdr.command = SLI4_MBOX_COMMAND_READ_NVPARMS; 1042 1043 return sizeof(sli4_cmd_read_nvparms_t); 1044 } 1045 1046 /** 1047 * @brief Write a WRITE_NVPARMS command to the provided buffer. 1048 * 1049 * @param sli4 SLI context pointer. 1050 * @param buf Virtual pointer to the destination buffer. 1051 * @param size Buffer size, in bytes. 1052 * @param wwpn WWPN to write - pointer to array of 8 uint8_t. 1053 * @param wwnn WWNN to write - pointer to array of 8 uint8_t. 1054 * @param hard_alpa Hard ALPA to write. 1055 * @param preferred_d_id Preferred D_ID to write. 1056 * 1057 * @return Returns the number of bytes written. 1058 */ 1059 int32_t 1060 sli_cmd_write_nvparms(sli4_t *sli4, void *buf, size_t size, uint8_t *wwpn, uint8_t *wwnn, uint8_t hard_alpa, 1061 uint32_t preferred_d_id) 1062 { 1063 sli4_cmd_write_nvparms_t *write_nvparms = buf; 1064 1065 ocs_memset(buf, 0, size); 1066 1067 write_nvparms->hdr.command = SLI4_MBOX_COMMAND_WRITE_NVPARMS; 1068 ocs_memcpy(write_nvparms->wwpn, wwpn, 8); 1069 ocs_memcpy(write_nvparms->wwnn, wwnn, 8); 1070 write_nvparms->hard_alpa = hard_alpa; 1071 write_nvparms->preferred_d_id = preferred_d_id; 1072 1073 return sizeof(sli4_cmd_write_nvparms_t); 1074 } 1075 1076 /** 1077 * @brief Write a READ_REV command to the provided buffer. 1078 * 1079 * @param sli4 SLI context pointer. 1080 * @param buf Virtual pointer to the destination buffer. 1081 * @param size Buffer size, in bytes. 1082 * @param vpd Pointer to the buffer. 1083 * 1084 * @return Returns the number of bytes written. 1085 */ 1086 static int32_t 1087 sli_cmd_read_rev(sli4_t *sli4, void *buf, size_t size, ocs_dma_t *vpd) 1088 { 1089 sli4_cmd_read_rev_t *read_rev = buf; 1090 1091 ocs_memset(buf, 0, size); 1092 1093 read_rev->hdr.command = SLI4_MBOX_COMMAND_READ_REV; 1094 1095 if (vpd && vpd->size) { 1096 read_rev->vpd = TRUE; 1097 1098 read_rev->available_length = vpd->size; 1099 1100 read_rev->physical_address_low = ocs_addr32_lo(vpd->phys); 1101 read_rev->physical_address_high = ocs_addr32_hi(vpd->phys); 1102 } 1103 1104 return sizeof(sli4_cmd_read_rev_t); 1105 } 1106 1107 /** 1108 * @ingroup sli 1109 * @brief Write a READ_SPARM64 command to the provided buffer. 1110 * 1111 * @param sli4 SLI context pointer. 1112 * @param buf Virtual pointer to the destination buffer. 1113 * @param size Buffer size, in bytes. 1114 * @param dma DMA buffer for the service parameters. 1115 * @param vpi VPI used to determine the WWN. 1116 * 1117 * @return Returns the number of bytes written. 1118 */ 1119 int32_t 1120 sli_cmd_read_sparm64(sli4_t *sli4, void *buf, size_t size, ocs_dma_t *dma, 1121 uint16_t vpi) 1122 { 1123 sli4_cmd_read_sparm64_t *read_sparm64 = buf; 1124 1125 ocs_memset(buf, 0, size); 1126 1127 if (SLI4_READ_SPARM64_VPI_SPECIAL == vpi) { 1128 ocs_log_test(sli4->os, "special VPI not supported!!!\n"); 1129 return -1; 1130 } 1131 1132 if (!dma || !dma->phys) { 1133 ocs_log_test(sli4->os, "bad DMA buffer\n"); 1134 return -1; 1135 } 1136 1137 read_sparm64->hdr.command = SLI4_MBOX_COMMAND_READ_SPARM64; 1138 1139 read_sparm64->bde_64.bde_type = SLI4_BDE_TYPE_BDE_64; 1140 read_sparm64->bde_64.buffer_length = dma->size; 1141 read_sparm64->bde_64.u.data.buffer_address_low = ocs_addr32_lo(dma->phys); 1142 read_sparm64->bde_64.u.data.buffer_address_high = ocs_addr32_hi(dma->phys); 1143 1144 read_sparm64->vpi = vpi; 1145 1146 return sizeof(sli4_cmd_read_sparm64_t); 1147 } 1148 1149 /** 1150 * @ingroup sli 1151 * @brief Write a READ_TOPOLOGY command to the provided buffer. 1152 * 1153 * @param sli4 SLI context pointer. 1154 * @param buf Virtual pointer to the destination buffer. 1155 * @param size Buffer size, in bytes. 1156 * @param dma DMA buffer for loop map (optional). 1157 * 1158 * @return Returns the number of bytes written. 1159 */ 1160 int32_t 1161 sli_cmd_read_topology(sli4_t *sli4, void *buf, size_t size, ocs_dma_t *dma) 1162 { 1163 sli4_cmd_read_topology_t *read_topo = buf; 1164 1165 ocs_memset(buf, 0, size); 1166 1167 read_topo->hdr.command = SLI4_MBOX_COMMAND_READ_TOPOLOGY; 1168 1169 if (dma && dma->size) { 1170 if (dma->size < SLI4_MIN_LOOP_MAP_BYTES) { 1171 ocs_log_test(sli4->os, "loop map buffer too small %jd\n", 1172 dma->size); 1173 return 0; 1174 } 1175 1176 ocs_memset(dma->virt, 0, dma->size); 1177 1178 read_topo->bde_loop_map.bde_type = SLI4_BDE_TYPE_BDE_64; 1179 read_topo->bde_loop_map.buffer_length = dma->size; 1180 read_topo->bde_loop_map.u.data.buffer_address_low = ocs_addr32_lo(dma->phys); 1181 read_topo->bde_loop_map.u.data.buffer_address_high = ocs_addr32_hi(dma->phys); 1182 } 1183 1184 return sizeof(sli4_cmd_read_topology_t); 1185 } 1186 1187 /** 1188 * @ingroup sli 1189 * @brief Write a REG_FCFI command to the provided buffer. 1190 * 1191 * @param sli4 SLI context pointer. 1192 * @param buf Virtual pointer to the destination buffer. 1193 * @param size Buffer size, in bytes. 1194 * @param index FCF index returned by READ_FCF_TABLE. 1195 * @param rq_cfg RQ_ID/R_CTL/TYPE routing information 1196 * @param vlan_id VLAN ID tag. 1197 * 1198 * @return Returns the number of bytes written. 1199 */ 1200 int32_t 1201 sli_cmd_reg_fcfi(sli4_t *sli4, void *buf, size_t size, uint16_t index, sli4_cmd_rq_cfg_t rq_cfg[SLI4_CMD_REG_FCFI_NUM_RQ_CFG], uint16_t vlan_id) 1202 { 1203 sli4_cmd_reg_fcfi_t *reg_fcfi = buf; 1204 uint32_t i; 1205 1206 ocs_memset(buf, 0, size); 1207 1208 reg_fcfi->hdr.command = SLI4_MBOX_COMMAND_REG_FCFI; 1209 1210 reg_fcfi->fcf_index = index; 1211 1212 for (i = 0; i < SLI4_CMD_REG_FCFI_NUM_RQ_CFG; i++) { 1213 switch(i) { 1214 case 0: reg_fcfi->rq_id_0 = rq_cfg[0].rq_id; break; 1215 case 1: reg_fcfi->rq_id_1 = rq_cfg[1].rq_id; break; 1216 case 2: reg_fcfi->rq_id_2 = rq_cfg[2].rq_id; break; 1217 case 3: reg_fcfi->rq_id_3 = rq_cfg[3].rq_id; break; 1218 } 1219 reg_fcfi->rq_cfg[i].r_ctl_mask = rq_cfg[i].r_ctl_mask; 1220 reg_fcfi->rq_cfg[i].r_ctl_match = rq_cfg[i].r_ctl_match; 1221 reg_fcfi->rq_cfg[i].type_mask = rq_cfg[i].type_mask; 1222 reg_fcfi->rq_cfg[i].type_match = rq_cfg[i].type_match; 1223 } 1224 1225 if (vlan_id) { 1226 reg_fcfi->vv = TRUE; 1227 reg_fcfi->vlan_tag = vlan_id; 1228 } 1229 1230 return sizeof(sli4_cmd_reg_fcfi_t); 1231 } 1232 1233 /** 1234 * @brief Write REG_FCFI_MRQ to provided command buffer 1235 * 1236 * @param sli4 SLI context pointer. 1237 * @param buf Virtual pointer to the destination buffer. 1238 * @param size Buffer size, in bytes. 1239 * @param fcf_index FCF index returned by READ_FCF_TABLE. 1240 * @param vlan_id VLAN ID tag. 1241 * @param rr_quant Round robin quanta if RQ selection policy is 2 1242 * @param rq_selection_policy RQ selection policy 1243 * @param num_rqs Array of count of RQs per filter 1244 * @param rq_ids Array of RQ ids per filter 1245 * @param rq_cfg RQ_ID/R_CTL/TYPE routing information 1246 * 1247 * @return returns 0 for success, a negative error code value for failure. 1248 */ 1249 int32_t 1250 sli_cmd_reg_fcfi_mrq(sli4_t *sli4, void *buf, size_t size, uint8_t mode, 1251 uint16_t fcf_index, uint16_t vlan_id, uint8_t rq_selection_policy, 1252 uint8_t mrq_bit_mask, uint16_t num_mrqs, 1253 sli4_cmd_rq_cfg_t rq_cfg[SLI4_CMD_REG_FCFI_NUM_RQ_CFG]) 1254 { 1255 sli4_cmd_reg_fcfi_mrq_t *reg_fcfi_mrq = buf; 1256 uint32_t i; 1257 1258 ocs_memset(buf, 0, size); 1259 1260 reg_fcfi_mrq->hdr.command = SLI4_MBOX_COMMAND_REG_FCFI_MRQ; 1261 if (mode == SLI4_CMD_REG_FCFI_SET_FCFI_MODE) { 1262 reg_fcfi_mrq->fcf_index = fcf_index; 1263 if (vlan_id) { 1264 reg_fcfi_mrq->vv = TRUE; 1265 reg_fcfi_mrq->vlan_tag = vlan_id; 1266 } 1267 goto done; 1268 } 1269 1270 reg_fcfi_mrq->mode = mode; 1271 for (i = 0; i < SLI4_CMD_REG_FCFI_NUM_RQ_CFG; i++) { 1272 reg_fcfi_mrq->rq_cfg[i].r_ctl_mask = rq_cfg[i].r_ctl_mask; 1273 reg_fcfi_mrq->rq_cfg[i].r_ctl_match = rq_cfg[i].r_ctl_match; 1274 reg_fcfi_mrq->rq_cfg[i].type_mask = rq_cfg[i].type_mask; 1275 reg_fcfi_mrq->rq_cfg[i].type_match = rq_cfg[i].type_match; 1276 1277 switch(i) { 1278 case 3: reg_fcfi_mrq->rq_id_3 = rq_cfg[i].rq_id; break; 1279 case 2: reg_fcfi_mrq->rq_id_2 = rq_cfg[i].rq_id; break; 1280 case 1: reg_fcfi_mrq->rq_id_1 = rq_cfg[i].rq_id; break; 1281 case 0: reg_fcfi_mrq->rq_id_0 = rq_cfg[i].rq_id; break; 1282 } 1283 } 1284 1285 reg_fcfi_mrq->rq_selection_policy = rq_selection_policy; 1286 reg_fcfi_mrq->mrq_filter_bitmask = mrq_bit_mask; 1287 reg_fcfi_mrq->num_mrq_pairs = num_mrqs; 1288 done: 1289 return sizeof(sli4_cmd_reg_fcfi_mrq_t); 1290 } 1291 1292 /** 1293 * @ingroup sli 1294 * @brief Write a REG_RPI command to the provided buffer. 1295 * 1296 * @param sli4 SLI context pointer. 1297 * @param buf Virtual pointer to the destination buffer. 1298 * @param size Buffer size, in bytes. 1299 * @param nport_id Remote F/N_Port_ID. 1300 * @param rpi Previously-allocated Remote Port Indicator. 1301 * @param vpi Previously-allocated Virtual Port Indicator. 1302 * @param dma DMA buffer that contains the remote port's service parameters. 1303 * @param update Boolean indicating an update to an existing RPI (TRUE) 1304 * or a new registration (FALSE). 1305 * 1306 * @return Returns the number of bytes written. 1307 */ 1308 int32_t 1309 sli_cmd_reg_rpi(sli4_t *sli4, void *buf, size_t size, uint32_t nport_id, uint16_t rpi, 1310 uint16_t vpi, ocs_dma_t *dma, uint8_t update, uint8_t enable_t10_pi) 1311 { 1312 sli4_cmd_reg_rpi_t *reg_rpi = buf; 1313 1314 ocs_memset(buf, 0, size); 1315 1316 reg_rpi->hdr.command = SLI4_MBOX_COMMAND_REG_RPI; 1317 1318 reg_rpi->rpi = rpi; 1319 reg_rpi->remote_n_port_id = nport_id; 1320 reg_rpi->upd = update; 1321 reg_rpi->etow = enable_t10_pi; 1322 1323 reg_rpi->bde_64.bde_type = SLI4_BDE_TYPE_BDE_64; 1324 reg_rpi->bde_64.buffer_length = SLI4_REG_RPI_BUF_LEN; 1325 reg_rpi->bde_64.u.data.buffer_address_low = ocs_addr32_lo(dma->phys); 1326 reg_rpi->bde_64.u.data.buffer_address_high = ocs_addr32_hi(dma->phys); 1327 1328 reg_rpi->vpi = vpi; 1329 1330 return sizeof(sli4_cmd_reg_rpi_t); 1331 } 1332 1333 /** 1334 * @ingroup sli 1335 * @brief Write a REG_VFI command to the provided buffer. 1336 * 1337 * @param sli4 SLI context pointer. 1338 * @param buf Virtual pointer to the destination buffer. 1339 * @param size Buffer size, in bytes. 1340 * @param domain Pointer to the domain object. 1341 * 1342 * @return Returns the number of bytes written. 1343 */ 1344 int32_t 1345 sli_cmd_reg_vfi(sli4_t *sli4, void *buf, size_t size, ocs_domain_t *domain) 1346 { 1347 sli4_cmd_reg_vfi_t *reg_vfi = buf; 1348 1349 if (!sli4 || !buf || !domain) { 1350 return 0; 1351 } 1352 1353 ocs_memset(buf, 0, size); 1354 1355 reg_vfi->hdr.command = SLI4_MBOX_COMMAND_REG_VFI; 1356 1357 reg_vfi->vfi = domain->indicator; 1358 1359 reg_vfi->fcfi = domain->fcf_indicator; 1360 1361 /* TODO contents of domain->dma only valid if topo == FABRIC */ 1362 reg_vfi->sparm.bde_type = SLI4_BDE_TYPE_BDE_64; 1363 reg_vfi->sparm.buffer_length = 0x70; 1364 reg_vfi->sparm.u.data.buffer_address_low = ocs_addr32_lo(domain->dma.phys); 1365 reg_vfi->sparm.u.data.buffer_address_high = ocs_addr32_hi(domain->dma.phys); 1366 1367 reg_vfi->e_d_tov = sli4->config.e_d_tov; 1368 reg_vfi->r_a_tov = sli4->config.r_a_tov; 1369 1370 reg_vfi->vp = TRUE; 1371 reg_vfi->vpi = domain->sport->indicator; 1372 ocs_memcpy(reg_vfi->wwpn, &domain->sport->sli_wwpn, sizeof(reg_vfi->wwpn)); 1373 reg_vfi->local_n_port_id = domain->sport->fc_id; 1374 1375 return sizeof(sli4_cmd_reg_vfi_t); 1376 } 1377 1378 /** 1379 * @ingroup sli 1380 * @brief Write a REG_VPI command to the provided buffer. 1381 * 1382 * @param sli4 SLI context pointer. 1383 * @param buf Virtual pointer to the destination buffer. 1384 * @param size Buffer size, in bytes. 1385 * @param sport Point to SLI Port object. 1386 * @param update Boolean indicating whether to update the existing VPI (true) 1387 * or create a new VPI (false). 1388 * 1389 * @return Returns the number of bytes written. 1390 */ 1391 int32_t 1392 sli_cmd_reg_vpi(sli4_t *sli4, void *buf, size_t size, ocs_sli_port_t *sport, uint8_t update) 1393 { 1394 sli4_cmd_reg_vpi_t *reg_vpi = buf; 1395 1396 if (!sli4 || !buf || !sport) { 1397 return 0; 1398 } 1399 1400 ocs_memset(buf, 0, size); 1401 1402 reg_vpi->hdr.command = SLI4_MBOX_COMMAND_REG_VPI; 1403 1404 reg_vpi->local_n_port_id = sport->fc_id; 1405 reg_vpi->upd = update != 0; 1406 ocs_memcpy(reg_vpi->wwpn, &sport->sli_wwpn, sizeof(reg_vpi->wwpn)); 1407 reg_vpi->vpi = sport->indicator; 1408 reg_vpi->vfi = sport->domain->indicator; 1409 1410 return sizeof(sli4_cmd_reg_vpi_t); 1411 } 1412 1413 /** 1414 * @brief Write a REQUEST_FEATURES command to the provided buffer. 1415 * 1416 * @param sli4 SLI context pointer. 1417 * @param buf Virtual pointer to the destination buffer. 1418 * @param size Buffer size, in bytes. 1419 * @param mask Features to request. 1420 * @param query Use feature query mode (does not change FW). 1421 * 1422 * @return Returns the number of bytes written. 1423 */ 1424 static int32_t 1425 sli_cmd_request_features(sli4_t *sli4, void *buf, size_t size, sli4_features_t mask, uint8_t query) 1426 { 1427 sli4_cmd_request_features_t *features = buf; 1428 1429 ocs_memset(buf, 0, size); 1430 1431 features->hdr.command = SLI4_MBOX_COMMAND_REQUEST_FEATURES; 1432 1433 if (query) { 1434 features->qry = TRUE; 1435 } 1436 features->command.dword = mask.dword; 1437 1438 return sizeof(sli4_cmd_request_features_t); 1439 } 1440 1441 /** 1442 * @ingroup sli 1443 * @brief Write a SLI_CONFIG command to the provided buffer. 1444 * 1445 * @param sli4 SLI context pointer. 1446 * @param buf Virtual pointer to the destination buffer. 1447 * @param size Buffer size, in bytes. 1448 * @param length Length in bytes of attached command. 1449 * @param dma DMA buffer for non-embedded commands. 1450 * 1451 * @return Returns the number of bytes written. 1452 */ 1453 int32_t 1454 sli_cmd_sli_config(sli4_t *sli4, void *buf, size_t size, uint32_t length, ocs_dma_t *dma) 1455 { 1456 sli4_cmd_sli_config_t *sli_config = NULL; 1457 1458 if ((length > sizeof(sli_config->payload.embed)) && (dma == NULL)) { 1459 ocs_log_test(sli4->os, "length(%d) > payload(%ld)\n", 1460 length, sizeof(sli_config->payload.embed)); 1461 return -1; 1462 } 1463 1464 sli_config = buf; 1465 1466 ocs_memset(buf, 0, size); 1467 1468 sli_config->hdr.command = SLI4_MBOX_COMMAND_SLI_CONFIG; 1469 if (NULL == dma) { 1470 sli_config->emb = TRUE; 1471 sli_config->payload_length = length; 1472 } else { 1473 sli_config->emb = FALSE; 1474 1475 sli_config->pmd_count = 1; 1476 1477 sli_config->payload.mem.address_low = ocs_addr32_lo(dma->phys); 1478 sli_config->payload.mem.address_high = ocs_addr32_hi(dma->phys); 1479 sli_config->payload.mem.length = dma->size; 1480 sli_config->payload_length = dma->size; 1481 #if defined(OCS_INCLUDE_DEBUG) 1482 /* save pointer to DMA for BMBX dumping purposes */ 1483 sli4->bmbx_non_emb_pmd = dma; 1484 #endif 1485 } 1486 1487 return offsetof(sli4_cmd_sli_config_t, payload.embed); 1488 } 1489 1490 /** 1491 * @brief Initialize SLI Port control register. 1492 * 1493 * @param sli4 SLI context pointer. 1494 * @param endian Endian value to write. 1495 * 1496 * @return Returns 0 on success, or a negative error code value on failure. 1497 */ 1498 1499 static int32_t 1500 sli_sliport_control(sli4_t *sli4, uint32_t endian) 1501 { 1502 uint32_t iter; 1503 int32_t rc; 1504 1505 rc = -1; 1506 1507 /* Initialize port, endian */ 1508 sli_reg_write(sli4, SLI4_REG_SLIPORT_CONTROL, endian | SLI4_SLIPORT_CONTROL_IP); 1509 1510 for (iter = 0; iter < 3000; iter ++) { 1511 ocs_udelay(SLI4_INIT_PORT_DELAY_US); 1512 if (sli_fw_ready(sli4) == 1) { 1513 rc = 0; 1514 break; 1515 } 1516 } 1517 1518 if (rc != 0) { 1519 ocs_log_crit(sli4->os, "port failed to become ready after initialization\n"); 1520 } 1521 1522 return rc; 1523 } 1524 1525 /** 1526 * @ingroup sli 1527 * @brief Write a UNREG_FCFI command to the provided buffer. 1528 * 1529 * @param sli4 SLI context pointer. 1530 * @param buf Virtual pointer to the destination buffer. 1531 * @param size Buffer size, in bytes. 1532 * @param indicator Indicator value. 1533 * 1534 * @return Returns the number of bytes written. 1535 */ 1536 int32_t 1537 sli_cmd_unreg_fcfi(sli4_t *sli4, void *buf, size_t size, uint16_t indicator) 1538 { 1539 sli4_cmd_unreg_fcfi_t *unreg_fcfi = buf; 1540 1541 if (!sli4 || !buf) { 1542 return 0; 1543 } 1544 1545 ocs_memset(buf, 0, size); 1546 1547 unreg_fcfi->hdr.command = SLI4_MBOX_COMMAND_UNREG_FCFI; 1548 1549 unreg_fcfi->fcfi = indicator; 1550 1551 return sizeof(sli4_cmd_unreg_fcfi_t); 1552 } 1553 1554 /** 1555 * @ingroup sli 1556 * @brief Write an UNREG_RPI command to the provided buffer. 1557 * 1558 * @param sli4 SLI context pointer. 1559 * @param buf Virtual pointer to the destination buffer. 1560 * @param size Buffer size, in bytes. 1561 * @param indicator Indicator value. 1562 * @param which Type of unregister, such as node, port, domain, or FCF. 1563 * @param fc_id FC address. 1564 * 1565 * @return Returns the number of bytes written. 1566 */ 1567 int32_t 1568 sli_cmd_unreg_rpi(sli4_t *sli4, void *buf, size_t size, uint16_t indicator, sli4_resource_e which, 1569 uint32_t fc_id) 1570 { 1571 sli4_cmd_unreg_rpi_t *unreg_rpi = buf; 1572 uint8_t index_indicator = 0; 1573 1574 if (!sli4 || !buf) { 1575 return 0; 1576 } 1577 1578 ocs_memset(buf, 0, size); 1579 1580 unreg_rpi->hdr.command = SLI4_MBOX_COMMAND_UNREG_RPI; 1581 1582 switch (which) { 1583 case SLI_RSRC_FCOE_RPI: 1584 index_indicator = SLI4_UNREG_RPI_II_RPI; 1585 if (fc_id != UINT32_MAX) { 1586 unreg_rpi->dp = TRUE; 1587 unreg_rpi->destination_n_port_id = fc_id & 0x00ffffff; 1588 } 1589 break; 1590 case SLI_RSRC_FCOE_VPI: 1591 index_indicator = SLI4_UNREG_RPI_II_VPI; 1592 break; 1593 case SLI_RSRC_FCOE_VFI: 1594 index_indicator = SLI4_UNREG_RPI_II_VFI; 1595 break; 1596 case SLI_RSRC_FCOE_FCFI: 1597 index_indicator = SLI4_UNREG_RPI_II_FCFI; 1598 break; 1599 default: 1600 ocs_log_test(sli4->os, "unknown type %#x\n", which); 1601 return 0; 1602 } 1603 1604 unreg_rpi->ii = index_indicator; 1605 unreg_rpi->index = indicator; 1606 1607 return sizeof(sli4_cmd_unreg_rpi_t); 1608 } 1609 1610 /** 1611 * @ingroup sli 1612 * @brief Write an UNREG_VFI command to the provided buffer. 1613 * 1614 * @param sli4 SLI context pointer. 1615 * @param buf Virtual pointer to the destination buffer. 1616 * @param size Buffer size, in bytes. 1617 * @param domain Pointer to the domain object 1618 * @param which Type of unregister, such as domain, FCFI, or everything. 1619 * 1620 * @return Returns the number of bytes written. 1621 */ 1622 int32_t 1623 sli_cmd_unreg_vfi(sli4_t *sli4, void *buf, size_t size, ocs_domain_t *domain, uint32_t which) 1624 { 1625 sli4_cmd_unreg_vfi_t *unreg_vfi = buf; 1626 1627 if (!sli4 || !buf || !domain) { 1628 return 0; 1629 } 1630 1631 ocs_memset(buf, 0, size); 1632 1633 unreg_vfi->hdr.command = SLI4_MBOX_COMMAND_UNREG_VFI; 1634 switch (which) { 1635 case SLI4_UNREG_TYPE_DOMAIN: 1636 unreg_vfi->index = domain->indicator; 1637 break; 1638 case SLI4_UNREG_TYPE_FCF: 1639 unreg_vfi->index = domain->fcf_indicator; 1640 break; 1641 case SLI4_UNREG_TYPE_ALL: 1642 unreg_vfi->index = UINT16_MAX; 1643 break; 1644 default: 1645 return 0; 1646 } 1647 1648 if (SLI4_UNREG_TYPE_DOMAIN != which) { 1649 unreg_vfi->ii = SLI4_UNREG_VFI_II_FCFI; 1650 } 1651 1652 return sizeof(sli4_cmd_unreg_vfi_t); 1653 } 1654 1655 /** 1656 * @ingroup sli 1657 * @brief Write an UNREG_VPI command to the provided buffer. 1658 * 1659 * @param sli4 SLI context pointer. 1660 * @param buf Virtual pointer to the destination buffer. 1661 * @param size Buffer size, in bytes. 1662 * @param indicator Indicator value. 1663 * @param which Type of unregister: port, domain, FCFI, everything 1664 * 1665 * @return Returns the number of bytes written. 1666 */ 1667 int32_t 1668 sli_cmd_unreg_vpi(sli4_t *sli4, void *buf, size_t size, uint16_t indicator, uint32_t which) 1669 { 1670 sli4_cmd_unreg_vpi_t *unreg_vpi = buf; 1671 1672 if (!sli4 || !buf) { 1673 return 0; 1674 } 1675 1676 ocs_memset(buf, 0, size); 1677 1678 unreg_vpi->hdr.command = SLI4_MBOX_COMMAND_UNREG_VPI; 1679 unreg_vpi->index = indicator; 1680 switch (which) { 1681 case SLI4_UNREG_TYPE_PORT: 1682 unreg_vpi->ii = SLI4_UNREG_VPI_II_VPI; 1683 break; 1684 case SLI4_UNREG_TYPE_DOMAIN: 1685 unreg_vpi->ii = SLI4_UNREG_VPI_II_VFI; 1686 break; 1687 case SLI4_UNREG_TYPE_FCF: 1688 unreg_vpi->ii = SLI4_UNREG_VPI_II_FCFI; 1689 break; 1690 case SLI4_UNREG_TYPE_ALL: 1691 unreg_vpi->index = UINT16_MAX; /* override indicator */ 1692 unreg_vpi->ii = SLI4_UNREG_VPI_II_FCFI; 1693 break; 1694 default: 1695 return 0; 1696 } 1697 1698 return sizeof(sli4_cmd_unreg_vpi_t); 1699 } 1700 1701 /** 1702 * @ingroup sli 1703 * @brief Write an CONFIG_AUTO_XFER_RDY command to the provided buffer. 1704 * 1705 * @param sli4 SLI context pointer. 1706 * @param buf Virtual pointer to the destination buffer. 1707 * @param size Buffer size, in bytes. 1708 * @param max_burst_len if the write FCP_DL is less than this size, 1709 * then the SLI port will generate the auto XFER_RDY. 1710 * 1711 * @return Returns the number of bytes written. 1712 */ 1713 int32_t 1714 sli_cmd_config_auto_xfer_rdy(sli4_t *sli4, void *buf, size_t size, uint32_t max_burst_len) 1715 { 1716 sli4_cmd_config_auto_xfer_rdy_t *req = buf; 1717 1718 if (!sli4 || !buf) { 1719 return 0; 1720 } 1721 1722 ocs_memset(buf, 0, size); 1723 1724 req->hdr.command = SLI4_MBOX_COMMAND_CONFIG_AUTO_XFER_RDY; 1725 req->max_burst_len = max_burst_len; 1726 1727 return sizeof(sli4_cmd_config_auto_xfer_rdy_t); 1728 } 1729 1730 /** 1731 * @ingroup sli 1732 * @brief Write an CONFIG_AUTO_XFER_RDY_HP command to the provided buffer. 1733 * 1734 * @param sli4 SLI context pointer. 1735 * @param buf Virtual pointer to the destination buffer. 1736 * @param size Buffer size, in bytes. 1737 * @param max_burst_len if the write FCP_DL is less than this size, 1738 * @param esoc enable start offset computation, 1739 * @param block_size block size, 1740 * then the SLI port will generate the auto XFER_RDY. 1741 * 1742 * @return Returns the number of bytes written. 1743 */ 1744 int32_t 1745 sli_cmd_config_auto_xfer_rdy_hp(sli4_t *sli4, void *buf, size_t size, uint32_t max_burst_len, 1746 uint32_t esoc, uint32_t block_size ) 1747 { 1748 sli4_cmd_config_auto_xfer_rdy_hp_t *req = buf; 1749 1750 if (!sli4 || !buf) { 1751 return 0; 1752 } 1753 1754 ocs_memset(buf, 0, size); 1755 1756 req->hdr.command = SLI4_MBOX_COMMAND_CONFIG_AUTO_XFER_RDY_HP; 1757 req->max_burst_len = max_burst_len; 1758 req->esoc = esoc; 1759 req->block_size = block_size; 1760 return sizeof(sli4_cmd_config_auto_xfer_rdy_hp_t); 1761 } 1762 1763 /** 1764 * @brief Write a COMMON_FUNCTION_RESET command. 1765 * 1766 * @param sli4 SLI context. 1767 * @param buf Destination buffer for the command. 1768 * @param size Buffer size, in bytes. 1769 * 1770 * @return Returns the number of bytes written. 1771 */ 1772 static int32_t 1773 sli_cmd_common_function_reset(sli4_t *sli4, void *buf, size_t size) 1774 { 1775 sli4_req_common_function_reset_t *reset = NULL; 1776 uint32_t sli_config_off = 0; 1777 1778 if (SLI4_PORT_TYPE_FC == sli4->port_type) { 1779 uint32_t payload_size; 1780 1781 /* Payload length must accommodate both request and response */ 1782 payload_size = max(sizeof(sli4_req_common_function_reset_t), 1783 sizeof(sli4_res_common_function_reset_t)); 1784 1785 sli_config_off = sli_cmd_sli_config(sli4, buf, size, payload_size, 1786 NULL); 1787 } 1788 reset = (sli4_req_common_function_reset_t *)((uint8_t *)buf + sli_config_off); 1789 1790 reset->hdr.opcode = SLI4_OPC_COMMON_FUNCTION_RESET; 1791 reset->hdr.subsystem = SLI4_SUBSYSTEM_COMMON; 1792 1793 return(sli_config_off + sizeof(sli4_req_common_function_reset_t)); 1794 } 1795 1796 /** 1797 * @brief Write a COMMON_CREATE_CQ command. 1798 * 1799 * @param sli4 SLI context. 1800 * @param buf Destination buffer for the command. 1801 * @param size Buffer size, in bytes. 1802 * @param qmem DMA memory for the queue. 1803 * @param eq_id Associated EQ_ID 1804 * @param ignored This parameter carries the ULP which is only used for WQ and RQs 1805 * 1806 * @note This creates a Version 0 message. 1807 * 1808 * @return Returns the number of bytes written. 1809 */ 1810 static int32_t 1811 sli_cmd_common_create_cq(sli4_t *sli4, void *buf, size_t size, 1812 ocs_dma_t *qmem, uint16_t eq_id, uint16_t ignored) 1813 { 1814 sli4_req_common_create_cq_v0_t *cqv0 = NULL; 1815 sli4_req_common_create_cq_v2_t *cqv2 = NULL; 1816 uint32_t sli_config_off = 0; 1817 uint32_t p; 1818 uintptr_t addr; 1819 uint32_t if_type = sli4->if_type; 1820 uint32_t page_bytes = 0; 1821 uint32_t num_pages = 0; 1822 uint32_t cmd_size = 0; 1823 uint32_t page_size = 0; 1824 uint32_t n_cqe = 0; 1825 1826 /* First calculate number of pages and the mailbox cmd length */ 1827 switch (if_type) 1828 { 1829 case SLI4_IF_TYPE_BE3_SKH_PF: 1830 page_bytes = SLI_PAGE_SIZE; 1831 num_pages = sli_page_count(qmem->size, page_bytes); 1832 cmd_size = sizeof(sli4_req_common_create_cq_v0_t) + (8 * num_pages); 1833 break; 1834 case SLI4_IF_TYPE_LANCER_FC_ETH: 1835 case SLI4_IF_TYPE_LANCER_G7: 1836 n_cqe = qmem->size / SLI4_CQE_BYTES; 1837 switch (n_cqe) { 1838 case 256: 1839 case 512: 1840 case 1024: 1841 case 2048: 1842 page_size = 1; 1843 break; 1844 case 4096: 1845 page_size = 2; 1846 break; 1847 default: 1848 return 0; 1849 } 1850 page_bytes = page_size * SLI_PAGE_SIZE; 1851 num_pages = sli_page_count(qmem->size, page_bytes); 1852 cmd_size = sizeof(sli4_req_common_create_cq_v2_t) + (8 * num_pages); 1853 break; 1854 default: 1855 ocs_log_test(sli4->os, "unsupported IF_TYPE %d\n", if_type); 1856 return -1; 1857 } 1858 1859 /* now that we have the mailbox command size, we can set SLI_CONFIG fields */ 1860 if (SLI4_PORT_TYPE_FC == sli4->port_type) { 1861 uint32_t payload_size; 1862 1863 /* Payload length must accommodate both request and response */ 1864 payload_size = max((size_t)cmd_size, sizeof(sli4_res_common_create_queue_t)); 1865 1866 sli_config_off = sli_cmd_sli_config(sli4, buf, size, payload_size, 1867 NULL); 1868 } 1869 1870 switch (if_type) 1871 { 1872 case SLI4_IF_TYPE_BE3_SKH_PF: 1873 cqv0 = (sli4_req_common_create_cq_v0_t *)((uint8_t *)buf + sli_config_off); 1874 cqv0->hdr.opcode = SLI4_OPC_COMMON_CREATE_CQ; 1875 cqv0->hdr.subsystem = SLI4_SUBSYSTEM_COMMON; 1876 cqv0->hdr.version = 0; 1877 cqv0->hdr.request_length = cmd_size - sizeof(sli4_req_hdr_t); 1878 1879 /* valid values for number of pages: 1, 2, 4 (sec 4.4.3) */ 1880 cqv0->num_pages = num_pages; 1881 switch (cqv0->num_pages) { 1882 case 1: 1883 cqv0->cqecnt = SLI4_CQ_CNT_256; 1884 break; 1885 case 2: 1886 cqv0->cqecnt = SLI4_CQ_CNT_512; 1887 break; 1888 case 4: 1889 cqv0->cqecnt = SLI4_CQ_CNT_1024; 1890 break; 1891 default: 1892 ocs_log_test(sli4->os, "num_pages %d not valid\n", cqv0->num_pages); 1893 return -1; 1894 } 1895 cqv0->evt = TRUE; 1896 cqv0->valid = TRUE; 1897 /* TODO cq->nodelay = ???; */ 1898 /* TODO cq->clswm = ???; */ 1899 cqv0->arm = FALSE; 1900 cqv0->eq_id = eq_id; 1901 1902 for (p = 0, addr = qmem->phys; 1903 p < cqv0->num_pages; 1904 p++, addr += page_bytes) { 1905 cqv0->page_physical_address[p].low = ocs_addr32_lo(addr); 1906 cqv0->page_physical_address[p].high = ocs_addr32_hi(addr); 1907 } 1908 1909 break; 1910 case SLI4_IF_TYPE_LANCER_FC_ETH: 1911 case SLI4_IF_TYPE_LANCER_G7: 1912 { 1913 cqv2 = (sli4_req_common_create_cq_v2_t *)((uint8_t *)buf + sli_config_off); 1914 cqv2->hdr.opcode = SLI4_OPC_COMMON_CREATE_CQ; 1915 cqv2->hdr.subsystem = SLI4_SUBSYSTEM_COMMON; 1916 cqv2->hdr.version = 2; 1917 cqv2->hdr.request_length = cmd_size - sizeof(sli4_req_hdr_t); 1918 1919 if (if_type == SLI4_IF_TYPE_LANCER_G7) 1920 cqv2->autovalid = TRUE; 1921 1922 cqv2->page_size = page_size; 1923 1924 /* valid values for number of pages: 1, 2, 4, 8 (sec 4.4.3) */ 1925 cqv2->num_pages = num_pages; 1926 if (!cqv2->num_pages || (cqv2->num_pages > SLI4_COMMON_CREATE_CQ_V2_MAX_PAGES)) { 1927 return 0; 1928 } 1929 1930 switch (cqv2->num_pages) { 1931 case 1: 1932 cqv2->cqecnt = SLI4_CQ_CNT_256; 1933 break; 1934 case 2: 1935 cqv2->cqecnt = SLI4_CQ_CNT_512; 1936 break; 1937 case 4: 1938 cqv2->cqecnt = SLI4_CQ_CNT_1024; 1939 break; 1940 case 8: 1941 cqv2->cqecnt = SLI4_CQ_CNT_LARGE; 1942 cqv2->cqe_count = n_cqe; 1943 break; 1944 default: 1945 ocs_log_test(sli4->os, "num_pages %d not valid\n", cqv2->num_pages); 1946 return -1; 1947 } 1948 1949 cqv2->evt = TRUE; 1950 cqv2->valid = TRUE; 1951 /* TODO cq->nodelay = ???; */ 1952 /* TODO cq->clswm = ???; */ 1953 cqv2->arm = FALSE; 1954 cqv2->eq_id = eq_id; 1955 1956 for (p = 0, addr = qmem->phys; 1957 p < cqv2->num_pages; 1958 p++, addr += page_bytes) { 1959 cqv2->page_physical_address[p].low = ocs_addr32_lo(addr); 1960 cqv2->page_physical_address[p].high = ocs_addr32_hi(addr); 1961 } 1962 } 1963 break; 1964 } 1965 1966 return (sli_config_off + cmd_size); 1967 } 1968 1969 /** 1970 * @brief Write a COMMON_DESTROY_CQ command. 1971 * 1972 * @param sli4 SLI context. 1973 * @param buf Destination buffer for the command. 1974 * @param size Buffer size, in bytes. 1975 * @param cq_id CQ ID 1976 * 1977 * @note This creates a Version 0 message. 1978 * 1979 * @return Returns the number of bytes written. 1980 */ 1981 static int32_t 1982 sli_cmd_common_destroy_cq(sli4_t *sli4, void *buf, size_t size, uint16_t cq_id) 1983 { 1984 sli4_req_common_destroy_cq_t *cq = NULL; 1985 uint32_t sli_config_off = 0; 1986 1987 if (SLI4_PORT_TYPE_FC == sli4->port_type) { 1988 sli_config_off = sli_cmd_sli_config(sli4, buf, size, 1989 /* Payload length must accommodate both request and response */ 1990 max(sizeof(sli4_req_common_destroy_cq_t), 1991 sizeof(sli4_res_hdr_t)), 1992 NULL); 1993 } 1994 cq = (sli4_req_common_destroy_cq_t *)((uint8_t *)buf + sli_config_off); 1995 1996 cq->hdr.opcode = SLI4_OPC_COMMON_DESTROY_CQ; 1997 cq->hdr.subsystem = SLI4_SUBSYSTEM_COMMON; 1998 cq->hdr.request_length = sizeof(sli4_req_common_destroy_cq_t) - 1999 sizeof(sli4_req_hdr_t); 2000 cq->cq_id = cq_id; 2001 2002 return(sli_config_off + sizeof(sli4_req_common_destroy_cq_t)); 2003 } 2004 2005 /** 2006 * @brief Write a COMMON_MODIFY_EQ_DELAY command. 2007 * 2008 * @param sli4 SLI context. 2009 * @param buf Destination buffer for the command. 2010 * @param size Buffer size, in bytes. 2011 * @param q Queue object array. 2012 * @param num_q Queue object array count. 2013 * @param shift Phase shift for staggering interrupts. 2014 * @param delay_mult Delay multiplier for limiting interrupt frequency. 2015 * 2016 * @return Returns the number of bytes written. 2017 */ 2018 static int32_t 2019 sli_cmd_common_modify_eq_delay(sli4_t *sli4, void *buf, size_t size, sli4_queue_t *q, int num_q, uint32_t shift, 2020 uint32_t delay_mult) 2021 { 2022 sli4_req_common_modify_eq_delay_t *modify_delay = NULL; 2023 uint32_t sli_config_off = 0; 2024 int i; 2025 2026 if (SLI4_PORT_TYPE_FC == sli4->port_type) { 2027 sli_config_off = sli_cmd_sli_config(sli4, buf, size, 2028 /* Payload length must accommodate both request and response */ 2029 max(sizeof(sli4_req_common_modify_eq_delay_t), sizeof(sli4_res_hdr_t)), 2030 NULL); 2031 } 2032 2033 modify_delay = (sli4_req_common_modify_eq_delay_t *)((uint8_t *)buf + sli_config_off); 2034 2035 modify_delay->hdr.opcode = SLI4_OPC_COMMON_MODIFY_EQ_DELAY; 2036 modify_delay->hdr.subsystem = SLI4_SUBSYSTEM_COMMON; 2037 modify_delay->hdr.request_length = sizeof(sli4_req_common_modify_eq_delay_t) - 2038 sizeof(sli4_req_hdr_t); 2039 2040 modify_delay->num_eq = num_q; 2041 2042 for (i = 0; i<num_q; i++) { 2043 modify_delay->eq_delay_record[i].eq_id = q[i].id; 2044 modify_delay->eq_delay_record[i].phase = shift; 2045 modify_delay->eq_delay_record[i].delay_multiplier = delay_mult; 2046 } 2047 2048 return(sli_config_off + sizeof(sli4_req_common_modify_eq_delay_t)); 2049 } 2050 2051 /** 2052 * @brief Write a COMMON_CREATE_EQ command. 2053 * 2054 * @param sli4 SLI context. 2055 * @param buf Destination buffer for the command. 2056 * @param size Buffer size, in bytes. 2057 * @param qmem DMA memory for the queue. 2058 * @param ignored1 Ignored (used for consistency among queue creation functions). 2059 * @param ignored2 Ignored (used for consistency among queue creation functions). 2060 * 2061 * @note Other queue creation routines use the last parameter to pass in 2062 * the associated Q_ID and ULP. EQ doesn't have an associated queue or ULP, 2063 * so these parameters are ignored 2064 * 2065 * @note This creates a Version 0 message 2066 * 2067 * @return Returns the number of bytes written. 2068 */ 2069 static int32_t 2070 sli_cmd_common_create_eq(sli4_t *sli4, void *buf, size_t size, ocs_dma_t *qmem, 2071 uint16_t ignored1, uint16_t ignored2) 2072 { 2073 sli4_req_common_create_eq_t *eq = NULL; 2074 uint32_t sli_config_off = 0; 2075 uint32_t p; 2076 uintptr_t addr; 2077 2078 if (SLI4_PORT_TYPE_FC == sli4->port_type) { 2079 uint32_t payload_size; 2080 2081 /* Payload length must accommodate both request and response */ 2082 payload_size = max(sizeof(sli4_req_common_create_eq_t), 2083 sizeof(sli4_res_common_create_queue_t)); 2084 2085 sli_config_off = sli_cmd_sli_config(sli4, buf, size, payload_size, 2086 NULL); 2087 } 2088 eq = (sli4_req_common_create_eq_t *)((uint8_t *)buf + sli_config_off); 2089 2090 eq->hdr.opcode = SLI4_OPC_COMMON_CREATE_EQ; 2091 eq->hdr.subsystem = SLI4_SUBSYSTEM_COMMON; 2092 eq->hdr.request_length = sizeof(sli4_req_common_create_eq_t) - 2093 sizeof(sli4_req_hdr_t); 2094 if (sli4->if_type == SLI4_IF_TYPE_LANCER_G7) { 2095 eq->hdr.version = 2; 2096 eq->autovalid = TRUE; 2097 } 2098 /* valid values for number of pages: 1, 2, 4 (sec 4.4.3) */ 2099 eq->num_pages = qmem->size / SLI_PAGE_SIZE; 2100 switch (eq->num_pages) { 2101 case 1: 2102 eq->eqesz = SLI4_EQE_SIZE_4; 2103 eq->count = SLI4_EQ_CNT_1024; 2104 break; 2105 case 2: 2106 eq->eqesz = SLI4_EQE_SIZE_4; 2107 eq->count = SLI4_EQ_CNT_2048; 2108 break; 2109 case 4: 2110 eq->eqesz = SLI4_EQE_SIZE_4; 2111 eq->count = SLI4_EQ_CNT_4096; 2112 break; 2113 default: 2114 ocs_log_test(sli4->os, "num_pages %d not valid\n", eq->num_pages); 2115 return -1; 2116 } 2117 eq->valid = TRUE; 2118 eq->arm = FALSE; 2119 eq->delay_multiplier = 32; 2120 2121 for (p = 0, addr = qmem->phys; 2122 p < eq->num_pages; 2123 p++, addr += SLI_PAGE_SIZE) { 2124 eq->page_address[p].low = ocs_addr32_lo(addr); 2125 eq->page_address[p].high = ocs_addr32_hi(addr); 2126 } 2127 2128 return(sli_config_off + sizeof(sli4_req_common_create_eq_t)); 2129 } 2130 2131 /** 2132 * @brief Write a COMMON_DESTROY_EQ command. 2133 * 2134 * @param sli4 SLI context. 2135 * @param buf Destination buffer for the command. 2136 * @param size Buffer size, in bytes. 2137 * @param eq_id Queue ID to destroy. 2138 * 2139 * @note Other queue creation routines use the last parameter to pass in 2140 * the associated Q_ID. EQ doesn't have an associated queue so this 2141 * parameter is ignored. 2142 * 2143 * @note This creates a Version 0 message. 2144 * 2145 * @return Returns the number of bytes written. 2146 */ 2147 static int32_t 2148 sli_cmd_common_destroy_eq(sli4_t *sli4, void *buf, size_t size, uint16_t eq_id) 2149 { 2150 sli4_req_common_destroy_eq_t *eq = NULL; 2151 uint32_t sli_config_off = 0; 2152 2153 if (SLI4_PORT_TYPE_FC == sli4->port_type) { 2154 sli_config_off = sli_cmd_sli_config(sli4, buf, size, 2155 /* Payload length must accommodate both request and response */ 2156 max(sizeof(sli4_req_common_destroy_eq_t), 2157 sizeof(sli4_res_hdr_t)), 2158 NULL); 2159 } 2160 eq = (sli4_req_common_destroy_eq_t *)((uint8_t *)buf + sli_config_off); 2161 2162 eq->hdr.opcode = SLI4_OPC_COMMON_DESTROY_EQ; 2163 eq->hdr.subsystem = SLI4_SUBSYSTEM_COMMON; 2164 eq->hdr.request_length = sizeof(sli4_req_common_destroy_eq_t) - 2165 sizeof(sli4_req_hdr_t); 2166 2167 eq->eq_id = eq_id; 2168 2169 return(sli_config_off + sizeof(sli4_req_common_destroy_eq_t)); 2170 } 2171 2172 /** 2173 * @brief Write a LOWLEVEL_SET_WATCHDOG command. 2174 * 2175 * @param sli4 SLI context. 2176 * @param buf Destination buffer for the command. 2177 * @param size Buffer size, in bytes. 2178 * @param timeout watchdog timer timeout in seconds 2179 * 2180 * @return void 2181 */ 2182 void 2183 sli4_cmd_lowlevel_set_watchdog(sli4_t *sli4, void *buf, size_t size, uint16_t timeout) 2184 { 2185 2186 sli4_req_lowlevel_set_watchdog_t *req = NULL; 2187 uint32_t sli_config_off = 0; 2188 2189 if (SLI4_PORT_TYPE_FC == sli4->port_type) { 2190 sli_config_off = sli_cmd_sli_config(sli4, buf, size, 2191 /* Payload length must accommodate both request and response */ 2192 max(sizeof(sli4_req_lowlevel_set_watchdog_t), 2193 sizeof(sli4_res_lowlevel_set_watchdog_t)), 2194 NULL); 2195 } 2196 req = (sli4_req_lowlevel_set_watchdog_t *)((uint8_t *)buf + sli_config_off); 2197 2198 req->hdr.opcode = SLI4_OPC_LOWLEVEL_SET_WATCHDOG; 2199 req->hdr.subsystem = SLI4_SUBSYSTEM_LOWLEVEL; 2200 req->hdr.request_length = sizeof(sli4_req_lowlevel_set_watchdog_t) - sizeof(sli4_req_hdr_t); 2201 req->watchdog_timeout = timeout; 2202 2203 return; 2204 } 2205 2206 static int32_t 2207 sli_cmd_common_get_cntl_attributes(sli4_t *sli4, void *buf, size_t size, ocs_dma_t *dma) 2208 { 2209 sli4_req_hdr_t *hdr = NULL; 2210 uint32_t sli_config_off = 0; 2211 2212 if (SLI4_PORT_TYPE_FC == sli4->port_type) { 2213 sli_config_off = sli_cmd_sli_config(sli4, buf, size, 2214 sizeof(sli4_req_hdr_t), 2215 dma); 2216 } 2217 2218 if (dma == NULL) { 2219 return 0; 2220 } 2221 2222 ocs_memset(dma->virt, 0, dma->size); 2223 2224 hdr = dma->virt; 2225 2226 hdr->opcode = SLI4_OPC_COMMON_GET_CNTL_ATTRIBUTES; 2227 hdr->subsystem = SLI4_SUBSYSTEM_COMMON; 2228 hdr->request_length = dma->size; 2229 2230 return(sli_config_off + sizeof(sli4_req_hdr_t)); 2231 } 2232 2233 /** 2234 * @brief Write a COMMON_GET_CNTL_ADDL_ATTRIBUTES command. 2235 * 2236 * @param sli4 SLI context. 2237 * @param buf Destination buffer for the command. 2238 * @param size Buffer size, in bytes. 2239 * @param dma DMA structure from which the data will be copied. 2240 * 2241 * @note This creates a Version 0 message. 2242 * 2243 * @return Returns the number of bytes written. 2244 */ 2245 static int32_t 2246 sli_cmd_common_get_cntl_addl_attributes(sli4_t *sli4, void *buf, size_t size, ocs_dma_t *dma) 2247 { 2248 sli4_req_hdr_t *hdr = NULL; 2249 uint32_t sli_config_off = 0; 2250 2251 if (SLI4_PORT_TYPE_FC == sli4->port_type) { 2252 sli_config_off = sli_cmd_sli_config(sli4, buf, size, sizeof(sli4_req_hdr_t), dma); 2253 } 2254 2255 if (dma == NULL) { 2256 return 0; 2257 } 2258 2259 ocs_memset(dma->virt, 0, dma->size); 2260 2261 hdr = dma->virt; 2262 2263 hdr->opcode = SLI4_OPC_COMMON_GET_CNTL_ADDL_ATTRIBUTES; 2264 hdr->subsystem = SLI4_SUBSYSTEM_COMMON; 2265 hdr->request_length = dma->size; 2266 2267 return(sli_config_off + sizeof(sli4_req_hdr_t)); 2268 } 2269 2270 /** 2271 * @brief Write a COMMON_CREATE_MQ_EXT command. 2272 * 2273 * @param sli4 SLI context. 2274 * @param buf Destination buffer for the command. 2275 * @param size Buffer size, in bytes. 2276 * @param qmem DMA memory for the queue. 2277 * @param cq_id Associated CQ_ID. 2278 * @param ignored This parameter carries the ULP which is only used for WQ and RQs 2279 * 2280 * @note This creates a Version 0 message. 2281 * 2282 * @return Returns the number of bytes written. 2283 */ 2284 static int32_t 2285 sli_cmd_common_create_mq_ext(sli4_t *sli4, void *buf, size_t size, 2286 ocs_dma_t *qmem, uint16_t cq_id, uint16_t ignored) 2287 { 2288 sli4_req_common_create_mq_ext_t *mq = NULL; 2289 uint32_t sli_config_off = 0; 2290 uint32_t p; 2291 uintptr_t addr; 2292 2293 if (SLI4_PORT_TYPE_FC == sli4->port_type) { 2294 uint32_t payload_size; 2295 2296 /* Payload length must accommodate both request and response */ 2297 payload_size = max(sizeof(sli4_req_common_create_mq_ext_t), 2298 sizeof(sli4_res_common_create_queue_t)); 2299 2300 sli_config_off = sli_cmd_sli_config(sli4, buf, size, payload_size, 2301 NULL); 2302 } 2303 mq = (sli4_req_common_create_mq_ext_t *)((uint8_t *)buf + sli_config_off); 2304 2305 mq->hdr.opcode = SLI4_OPC_COMMON_CREATE_MQ_EXT; 2306 mq->hdr.subsystem = SLI4_SUBSYSTEM_COMMON; 2307 mq->hdr.request_length = sizeof(sli4_req_common_create_mq_ext_t) - 2308 sizeof(sli4_req_hdr_t); 2309 /* valid values for number of pages: 1, 2, 4, 8 (sec 4.4.12) */ 2310 mq->num_pages = qmem->size / SLI_PAGE_SIZE; 2311 switch (mq->num_pages) { 2312 case 1: 2313 mq->ring_size = SLI4_MQE_SIZE_16; 2314 break; 2315 case 2: 2316 mq->ring_size = SLI4_MQE_SIZE_32; 2317 break; 2318 case 4: 2319 mq->ring_size = SLI4_MQE_SIZE_64; 2320 break; 2321 case 8: 2322 mq->ring_size = SLI4_MQE_SIZE_128; 2323 break; 2324 default: 2325 ocs_log_test(sli4->os, "num_pages %d not valid\n", mq->num_pages); 2326 return -1; 2327 } 2328 2329 /* TODO break this down by sli4->config.topology */ 2330 mq->async_event_bitmap = SLI4_ASYNC_EVT_FC_FCOE; 2331 2332 if (sli4->config.mq_create_version) { 2333 mq->cq_id_v1 = cq_id; 2334 mq->hdr.version = 1; 2335 } 2336 else { 2337 mq->cq_id_v0 = cq_id; 2338 } 2339 mq->val = TRUE; 2340 2341 for (p = 0, addr = qmem->phys; 2342 p < mq->num_pages; 2343 p++, addr += SLI_PAGE_SIZE) { 2344 mq->page_physical_address[p].low = ocs_addr32_lo(addr); 2345 mq->page_physical_address[p].high = ocs_addr32_hi(addr); 2346 } 2347 2348 return(sli_config_off + sizeof(sli4_req_common_create_mq_ext_t)); 2349 } 2350 2351 /** 2352 * @brief Write a COMMON_DESTROY_MQ command. 2353 * 2354 * @param sli4 SLI context. 2355 * @param buf Destination buffer for the command. 2356 * @param size Buffer size, in bytes. 2357 * @param mq_id MQ ID 2358 * 2359 * @note This creates a Version 0 message. 2360 * 2361 * @return Returns the number of bytes written. 2362 */ 2363 static int32_t 2364 sli_cmd_common_destroy_mq(sli4_t *sli4, void *buf, size_t size, uint16_t mq_id) 2365 { 2366 sli4_req_common_destroy_mq_t *mq = NULL; 2367 uint32_t sli_config_off = 0; 2368 2369 if (SLI4_PORT_TYPE_FC == sli4->port_type) { 2370 sli_config_off = sli_cmd_sli_config(sli4, buf, size, 2371 /* Payload length must accommodate both request and response */ 2372 max(sizeof(sli4_req_common_destroy_mq_t), 2373 sizeof(sli4_res_hdr_t)), 2374 NULL); 2375 } 2376 mq = (sli4_req_common_destroy_mq_t *)((uint8_t *)buf + sli_config_off); 2377 2378 mq->hdr.opcode = SLI4_OPC_COMMON_DESTROY_MQ; 2379 mq->hdr.subsystem = SLI4_SUBSYSTEM_COMMON; 2380 mq->hdr.request_length = sizeof(sli4_req_common_destroy_mq_t) - 2381 sizeof(sli4_req_hdr_t); 2382 2383 mq->mq_id = mq_id; 2384 2385 return(sli_config_off + sizeof(sli4_req_common_destroy_mq_t)); 2386 } 2387 2388 /** 2389 * @ingroup sli 2390 * @brief Write a COMMON_NOP command 2391 * 2392 * @param sli4 SLI context. 2393 * @param buf Destination buffer for the command. 2394 * @param size Buffer size, in bytes. 2395 * @param context NOP context value (passed to response, except on FC/FCoE). 2396 * 2397 * @return Returns the number of bytes written. 2398 */ 2399 int32_t 2400 sli_cmd_common_nop(sli4_t *sli4, void *buf, size_t size, uint64_t context) 2401 { 2402 sli4_req_common_nop_t *nop = NULL; 2403 uint32_t sli_config_off = 0; 2404 2405 if (SLI4_PORT_TYPE_FC == sli4->port_type) { 2406 sli_config_off = sli_cmd_sli_config(sli4, buf, size, 2407 /* Payload length must accommodate both request and response */ 2408 max(sizeof(sli4_req_common_nop_t), sizeof(sli4_res_common_nop_t)), 2409 NULL); 2410 } 2411 2412 nop = (sli4_req_common_nop_t *)((uint8_t *)buf + sli_config_off); 2413 2414 nop->hdr.opcode = SLI4_OPC_COMMON_NOP; 2415 nop->hdr.subsystem = SLI4_SUBSYSTEM_COMMON; 2416 nop->hdr.request_length = 8; 2417 2418 ocs_memcpy(&nop->context, &context, sizeof(context)); 2419 2420 return(sli_config_off + sizeof(sli4_req_common_nop_t)); 2421 } 2422 2423 /** 2424 * @ingroup sli 2425 * @brief Write a COMMON_GET_RESOURCE_EXTENT_INFO command. 2426 * 2427 * @param sli4 SLI context. 2428 * @param buf Destination buffer for the command. 2429 * @param size Buffer size, in bytes. 2430 * @param rtype Resource type (for example, XRI, VFI, VPI, and RPI). 2431 * 2432 * @return Returns the number of bytes written. 2433 */ 2434 int32_t 2435 sli_cmd_common_get_resource_extent_info(sli4_t *sli4, void *buf, size_t size, uint16_t rtype) 2436 { 2437 sli4_req_common_get_resource_extent_info_t *extent = NULL; 2438 uint32_t sli_config_off = 0; 2439 2440 if (SLI4_PORT_TYPE_FC == sli4->port_type) { 2441 sli_config_off = sli_cmd_sli_config(sli4, buf, size, 2442 sizeof(sli4_req_common_get_resource_extent_info_t), 2443 NULL); 2444 } 2445 2446 extent = (sli4_req_common_get_resource_extent_info_t *)((uint8_t *)buf + sli_config_off); 2447 2448 extent->hdr.opcode = SLI4_OPC_COMMON_GET_RESOURCE_EXTENT_INFO; 2449 extent->hdr.subsystem = SLI4_SUBSYSTEM_COMMON; 2450 extent->hdr.request_length = 4; 2451 2452 extent->resource_type = rtype; 2453 2454 return(sli_config_off + sizeof(sli4_req_common_get_resource_extent_info_t)); 2455 } 2456 2457 /** 2458 * @ingroup sli 2459 * @brief Write a COMMON_GET_SLI4_PARAMETERS command. 2460 * 2461 * @param sli4 SLI context. 2462 * @param buf Destination buffer for the command. 2463 * @param size Buffer size, in bytes. 2464 * 2465 * @return Returns the number of bytes written. 2466 */ 2467 int32_t 2468 sli_cmd_common_get_sli4_parameters(sli4_t *sli4, void *buf, size_t size) 2469 { 2470 sli4_req_hdr_t *hdr = NULL; 2471 uint32_t sli_config_off = 0; 2472 2473 if (SLI4_PORT_TYPE_FC == sli4->port_type) { 2474 sli_config_off = sli_cmd_sli_config(sli4, buf, size, 2475 sizeof(sli4_res_common_get_sli4_parameters_t), 2476 NULL); 2477 } 2478 2479 hdr = (sli4_req_hdr_t *)((uint8_t *)buf + sli_config_off); 2480 2481 hdr->opcode = SLI4_OPC_COMMON_GET_SLI4_PARAMETERS; 2482 hdr->subsystem = SLI4_SUBSYSTEM_COMMON; 2483 hdr->request_length = 0x50; 2484 2485 return(sli_config_off + sizeof(sli4_req_hdr_t)); 2486 } 2487 2488 /** 2489 * @brief Write a COMMON_QUERY_FW_CONFIG command to the provided buffer. 2490 * 2491 * @param sli4 SLI context pointer. 2492 * @param buf Virtual pointer to destination buffer. 2493 * @param size Buffer size in bytes. 2494 * 2495 * @return Returns the number of bytes written 2496 */ 2497 static int32_t 2498 sli_cmd_common_query_fw_config(sli4_t *sli4, void *buf, size_t size) 2499 { 2500 sli4_req_common_query_fw_config_t *fw_config; 2501 uint32_t sli_config_off = 0; 2502 uint32_t payload_size; 2503 2504 /* Payload length must accommodate both request and response */ 2505 payload_size = max(sizeof(sli4_req_common_query_fw_config_t), 2506 sizeof(sli4_res_common_query_fw_config_t)); 2507 2508 if (SLI4_PORT_TYPE_FC == sli4->port_type) { 2509 sli_config_off = sli_cmd_sli_config(sli4, buf, size, 2510 payload_size, 2511 NULL); 2512 } 2513 2514 fw_config = (sli4_req_common_query_fw_config_t*)((uint8_t*)buf + sli_config_off); 2515 fw_config->hdr.opcode = SLI4_OPC_COMMON_QUERY_FW_CONFIG; 2516 fw_config->hdr.subsystem = SLI4_SUBSYSTEM_COMMON; 2517 fw_config->hdr.request_length = payload_size - sizeof(sli4_req_hdr_t); 2518 return sli_config_off + sizeof(sli4_req_common_query_fw_config_t); 2519 } 2520 2521 /** 2522 * @brief Write a COMMON_GET_PORT_NAME command to the provided buffer. 2523 * 2524 * @param sli4 SLI context pointer. 2525 * @param buf Virtual pointer to destination buffer. 2526 * @param size Buffer size in bytes. 2527 * 2528 * @note Function supports both version 0 and 1 forms of this command via 2529 * the IF_TYPE. 2530 * 2531 * @return Returns the number of bytes written. 2532 */ 2533 static int32_t 2534 sli_cmd_common_get_port_name(sli4_t *sli4, void *buf, size_t size) 2535 { 2536 sli4_req_common_get_port_name_t *port_name; 2537 uint32_t sli_config_off = 0; 2538 uint32_t payload_size; 2539 uint8_t version = 0; 2540 uint8_t pt = 0; 2541 2542 /* Select command version according to IF_TYPE */ 2543 switch (sli4->if_type) { 2544 case SLI4_IF_TYPE_BE3_SKH_PF: 2545 case SLI4_IF_TYPE_BE3_SKH_VF: 2546 version = 0; 2547 break; 2548 case SLI4_IF_TYPE_LANCER_FC_ETH: 2549 case SLI4_IF_TYPE_LANCER_RDMA: 2550 case SLI4_IF_TYPE_LANCER_G7: 2551 version = 1; 2552 break; 2553 default: 2554 ocs_log_test(sli4->os, "unsupported IF_TYPE %d\n", sli4->if_type); 2555 return 0; 2556 } 2557 2558 /* Payload length must accommodate both request and response */ 2559 payload_size = max(sizeof(sli4_req_common_get_port_name_t), 2560 sizeof(sli4_res_common_get_port_name_t)); 2561 2562 if (SLI4_PORT_TYPE_FC == sli4->port_type) { 2563 sli_config_off = sli_cmd_sli_config(sli4, buf, size, 2564 payload_size, 2565 NULL); 2566 2567 pt = 1; 2568 } 2569 2570 port_name = (sli4_req_common_get_port_name_t *)((uint8_t *)buf + sli_config_off); 2571 2572 port_name->hdr.opcode = SLI4_OPC_COMMON_GET_PORT_NAME; 2573 port_name->hdr.subsystem = SLI4_SUBSYSTEM_COMMON; 2574 port_name->hdr.request_length = sizeof(sli4_req_hdr_t) + (version * sizeof(uint32_t)); 2575 port_name->hdr.version = version; 2576 2577 /* Set the port type value (ethernet=0, FC=1) for V1 commands */ 2578 if (version == 1) { 2579 port_name->pt = pt; 2580 } 2581 2582 return sli_config_off + port_name->hdr.request_length; 2583 } 2584 2585 /** 2586 * @ingroup sli 2587 * @brief Write a COMMON_WRITE_OBJECT command. 2588 * 2589 * @param sli4 SLI context. 2590 * @param buf Destination buffer for the command. 2591 * @param size Buffer size, in bytes. 2592 * @param noc True if the object should be written but not committed to flash. 2593 * @param eof True if this is the last write for this object. 2594 * @param desired_write_length Number of bytes of data to write to the object. 2595 * @param offset Offset, in bytes, from the start of the object. 2596 * @param object_name Name of the object to write. 2597 * @param dma DMA structure from which the data will be copied. 2598 * 2599 * @return Returns the number of bytes written. 2600 */ 2601 int32_t 2602 sli_cmd_common_write_object(sli4_t *sli4, void *buf, size_t size, 2603 uint16_t noc, uint16_t eof, uint32_t desired_write_length, 2604 uint32_t offset, 2605 char *object_name, 2606 ocs_dma_t *dma) 2607 { 2608 sli4_req_common_write_object_t *wr_obj = NULL; 2609 uint32_t sli_config_off = 0; 2610 sli4_bde_t *host_buffer; 2611 2612 if (SLI4_PORT_TYPE_FC == sli4->port_type) { 2613 sli_config_off = sli_cmd_sli_config(sli4, buf, size, 2614 sizeof (sli4_req_common_write_object_t) + sizeof (sli4_bde_t), 2615 NULL); 2616 } 2617 2618 wr_obj = (sli4_req_common_write_object_t *)((uint8_t *)buf + sli_config_off); 2619 2620 wr_obj->hdr.opcode = SLI4_OPC_COMMON_WRITE_OBJECT; 2621 wr_obj->hdr.subsystem = SLI4_SUBSYSTEM_COMMON; 2622 wr_obj->hdr.request_length = sizeof(*wr_obj) - 4*sizeof(uint32_t) + sizeof(sli4_bde_t); 2623 wr_obj->hdr.timeout = 0; 2624 wr_obj->hdr.version = 0; 2625 2626 wr_obj->noc = noc; 2627 wr_obj->eof = eof; 2628 wr_obj->desired_write_length = desired_write_length; 2629 wr_obj->write_offset = offset; 2630 ocs_strncpy(wr_obj->object_name, object_name, sizeof(wr_obj->object_name)); 2631 wr_obj->host_buffer_descriptor_count = 1; 2632 2633 host_buffer = (sli4_bde_t *)wr_obj->host_buffer_descriptor; 2634 2635 /* Setup to transfer xfer_size bytes to device */ 2636 host_buffer->bde_type = SLI4_BDE_TYPE_BDE_64; 2637 host_buffer->buffer_length = desired_write_length; 2638 host_buffer->u.data.buffer_address_low = ocs_addr32_lo(dma->phys); 2639 host_buffer->u.data.buffer_address_high = ocs_addr32_hi(dma->phys); 2640 2641 return(sli_config_off + sizeof(sli4_req_common_write_object_t) + sizeof (sli4_bde_t)); 2642 } 2643 2644 /** 2645 * @ingroup sli 2646 * @brief Write a COMMON_DELETE_OBJECT command. 2647 * 2648 * @param sli4 SLI context. 2649 * @param buf Destination buffer for the command. 2650 * @param size Buffer size, in bytes. 2651 * @param object_name Name of the object to write. 2652 * 2653 * @return Returns the number of bytes written. 2654 */ 2655 int32_t 2656 sli_cmd_common_delete_object(sli4_t *sli4, void *buf, size_t size, 2657 char *object_name) 2658 { 2659 sli4_req_common_delete_object_t *del_obj = NULL; 2660 uint32_t sli_config_off = 0; 2661 2662 if (SLI4_PORT_TYPE_FC == sli4->port_type) { 2663 sli_config_off = sli_cmd_sli_config(sli4, buf, size, 2664 sizeof (sli4_req_common_delete_object_t), 2665 NULL); 2666 } 2667 2668 del_obj = (sli4_req_common_delete_object_t *)((uint8_t *)buf + sli_config_off); 2669 2670 del_obj->hdr.opcode = SLI4_OPC_COMMON_DELETE_OBJECT; 2671 del_obj->hdr.subsystem = SLI4_SUBSYSTEM_COMMON; 2672 del_obj->hdr.request_length = sizeof(*del_obj); 2673 del_obj->hdr.timeout = 0; 2674 del_obj->hdr.version = 0; 2675 2676 ocs_strncpy(del_obj->object_name, object_name, sizeof(del_obj->object_name)); 2677 return(sli_config_off + sizeof(sli4_req_common_delete_object_t)); 2678 } 2679 2680 /** 2681 * @ingroup sli 2682 * @brief Write a COMMON_READ_OBJECT command. 2683 * 2684 * @param sli4 SLI context. 2685 * @param buf Destination buffer for the command. 2686 * @param size Buffer size, in bytes. 2687 * @param desired_read_length Number of bytes of data to read from the object. 2688 * @param offset Offset, in bytes, from the start of the object. 2689 * @param object_name Name of the object to read. 2690 * @param dma DMA structure from which the data will be copied. 2691 * 2692 * @return Returns the number of bytes written. 2693 */ 2694 int32_t 2695 sli_cmd_common_read_object(sli4_t *sli4, void *buf, size_t size, 2696 uint32_t desired_read_length, 2697 uint32_t offset, 2698 char *object_name, 2699 ocs_dma_t *dma) 2700 { 2701 sli4_req_common_read_object_t *rd_obj = NULL; 2702 uint32_t sli_config_off = 0; 2703 sli4_bde_t *host_buffer; 2704 2705 if (SLI4_PORT_TYPE_FC == sli4->port_type) { 2706 sli_config_off = sli_cmd_sli_config(sli4, buf, size, 2707 sizeof (sli4_req_common_read_object_t) + sizeof (sli4_bde_t), 2708 NULL); 2709 } 2710 2711 rd_obj = (sli4_req_common_read_object_t *)((uint8_t *)buf + sli_config_off); 2712 2713 rd_obj->hdr.opcode = SLI4_OPC_COMMON_READ_OBJECT; 2714 rd_obj->hdr.subsystem = SLI4_SUBSYSTEM_COMMON; 2715 rd_obj->hdr.request_length = sizeof(*rd_obj) - 4*sizeof(uint32_t) + sizeof(sli4_bde_t); 2716 rd_obj->hdr.timeout = 0; 2717 rd_obj->hdr.version = 0; 2718 2719 rd_obj->desired_read_length = desired_read_length; 2720 rd_obj->read_offset = offset; 2721 ocs_strncpy(rd_obj->object_name, object_name, sizeof(rd_obj->object_name)); 2722 rd_obj->host_buffer_descriptor_count = 1; 2723 2724 host_buffer = (sli4_bde_t *)rd_obj->host_buffer_descriptor; 2725 2726 /* Setup to transfer xfer_size bytes to device */ 2727 host_buffer->bde_type = SLI4_BDE_TYPE_BDE_64; 2728 host_buffer->buffer_length = desired_read_length; 2729 if (dma != NULL) { 2730 host_buffer->u.data.buffer_address_low = ocs_addr32_lo(dma->phys); 2731 host_buffer->u.data.buffer_address_high = ocs_addr32_hi(dma->phys); 2732 } else { 2733 host_buffer->u.data.buffer_address_low = 0; 2734 host_buffer->u.data.buffer_address_high = 0; 2735 } 2736 2737 return(sli_config_off + sizeof(sli4_req_common_read_object_t) + sizeof (sli4_bde_t)); 2738 } 2739 2740 /** 2741 * @ingroup sli 2742 * @brief Write a DMTF_EXEC_CLP_CMD command. 2743 * 2744 * @param sli4 SLI context. 2745 * @param buf Destination buffer for the command. 2746 * @param size Buffer size, in bytes. 2747 * @param cmd DMA structure that describes the buffer for the command. 2748 * @param resp DMA structure that describes the buffer for the response. 2749 * 2750 * @return Returns the number of bytes written. 2751 */ 2752 int32_t 2753 sli_cmd_dmtf_exec_clp_cmd(sli4_t *sli4, void *buf, size_t size, 2754 ocs_dma_t *cmd, 2755 ocs_dma_t *resp) 2756 { 2757 sli4_req_dmtf_exec_clp_cmd_t *clp_cmd = NULL; 2758 uint32_t sli_config_off = 0; 2759 2760 if (SLI4_PORT_TYPE_FC == sli4->port_type) { 2761 sli_config_off = sli_cmd_sli_config(sli4, buf, size, 2762 sizeof (sli4_req_dmtf_exec_clp_cmd_t), 2763 NULL); 2764 } 2765 2766 clp_cmd = (sli4_req_dmtf_exec_clp_cmd_t*)((uint8_t *)buf + sli_config_off); 2767 2768 clp_cmd->hdr.opcode = SLI4_OPC_DMTF_EXEC_CLP_CMD; 2769 clp_cmd->hdr.subsystem = SLI4_SUBSYSTEM_DMTF; 2770 clp_cmd->hdr.request_length = sizeof(sli4_req_dmtf_exec_clp_cmd_t) - 2771 sizeof(sli4_req_hdr_t); 2772 clp_cmd->hdr.timeout = 0; 2773 clp_cmd->hdr.version = 0; 2774 clp_cmd->cmd_buf_length = cmd->size; 2775 clp_cmd->cmd_buf_addr_low = ocs_addr32_lo(cmd->phys); 2776 clp_cmd->cmd_buf_addr_high = ocs_addr32_hi(cmd->phys); 2777 clp_cmd->resp_buf_length = resp->size; 2778 clp_cmd->resp_buf_addr_low = ocs_addr32_lo(resp->phys); 2779 clp_cmd->resp_buf_addr_high = ocs_addr32_hi(resp->phys); 2780 2781 return(sli_config_off + sizeof(sli4_req_dmtf_exec_clp_cmd_t)); 2782 } 2783 2784 /** 2785 * @ingroup sli 2786 * @brief Write a COMMON_SET_DUMP_LOCATION command. 2787 * 2788 * @param sli4 SLI context. 2789 * @param buf Destination buffer for the command. 2790 * @param size Buffer size, in bytes. 2791 * @param query Zero to set dump location, non-zero to query dump size 2792 * @param is_buffer_list Set to one if the buffer is a set of buffer descriptors or 2793 * set to 0 if the buffer is a contiguous dump area. 2794 * @param buffer DMA structure to which the dump will be copied. 2795 * 2796 * @return Returns the number of bytes written. 2797 */ 2798 int32_t 2799 sli_cmd_common_set_dump_location(sli4_t *sli4, void *buf, size_t size, 2800 uint8_t query, uint8_t is_buffer_list, 2801 ocs_dma_t *buffer, uint8_t fdb) 2802 { 2803 sli4_req_common_set_dump_location_t *set_dump_loc = NULL; 2804 uint32_t sli_config_off = 0; 2805 2806 if (SLI4_PORT_TYPE_FC == sli4->port_type) { 2807 sli_config_off = sli_cmd_sli_config(sli4, buf, size, 2808 sizeof (sli4_req_common_set_dump_location_t), 2809 NULL); 2810 } 2811 2812 set_dump_loc = (sli4_req_common_set_dump_location_t *)((uint8_t *)buf + sli_config_off); 2813 2814 set_dump_loc->hdr.opcode = SLI4_OPC_COMMON_SET_DUMP_LOCATION; 2815 set_dump_loc->hdr.subsystem = SLI4_SUBSYSTEM_COMMON; 2816 set_dump_loc->hdr.request_length = sizeof(sli4_req_common_set_dump_location_t) - sizeof(sli4_req_hdr_t); 2817 set_dump_loc->hdr.timeout = 0; 2818 set_dump_loc->hdr.version = 0; 2819 2820 set_dump_loc->blp = is_buffer_list; 2821 set_dump_loc->qry = query; 2822 set_dump_loc->fdb = fdb; 2823 2824 if (buffer) { 2825 set_dump_loc->buf_addr_low = ocs_addr32_lo(buffer->phys); 2826 set_dump_loc->buf_addr_high = ocs_addr32_hi(buffer->phys); 2827 set_dump_loc->buffer_length = buffer->len; 2828 } else { 2829 set_dump_loc->buf_addr_low = 0; 2830 set_dump_loc->buf_addr_high = 0; 2831 set_dump_loc->buffer_length = 0; 2832 } 2833 2834 return(sli_config_off + sizeof(sli4_req_common_set_dump_location_t)); 2835 } 2836 2837 /** 2838 * @ingroup sli 2839 * @brief Write a COMMON_SET_FEATURES command. 2840 * 2841 * @param sli4 SLI context. 2842 * @param buf Destination buffer for the command. 2843 * @param size Buffer size, in bytes. 2844 * @param feature Feature to set. 2845 * @param param_len Length of the parameter (must be a multiple of 4 bytes). 2846 * @param parameter Pointer to the parameter value. 2847 * 2848 * @return Returns the number of bytes written. 2849 */ 2850 int32_t 2851 sli_cmd_common_set_features(sli4_t *sli4, void *buf, size_t size, 2852 uint32_t feature, 2853 uint32_t param_len, 2854 void* parameter) 2855 { 2856 sli4_req_common_set_features_t *cmd = NULL; 2857 uint32_t sli_config_off = 0; 2858 2859 if (SLI4_PORT_TYPE_FC == sli4->port_type) { 2860 sli_config_off = sli_cmd_sli_config(sli4, buf, size, 2861 sizeof (sli4_req_common_set_features_t), 2862 NULL); 2863 } 2864 2865 cmd = (sli4_req_common_set_features_t *)((uint8_t *)buf + sli_config_off); 2866 2867 cmd->hdr.opcode = SLI4_OPC_COMMON_SET_FEATURES; 2868 cmd->hdr.subsystem = SLI4_SUBSYSTEM_COMMON; 2869 cmd->hdr.request_length = sizeof(sli4_req_common_set_features_t) - sizeof(sli4_req_hdr_t); 2870 cmd->hdr.timeout = 0; 2871 cmd->hdr.version = 0; 2872 2873 cmd->feature = feature; 2874 cmd->param_len = param_len; 2875 ocs_memcpy(cmd->params, parameter, param_len); 2876 2877 return(sli_config_off + sizeof(sli4_req_common_set_features_t)); 2878 } 2879 2880 /** 2881 * @ingroup sli 2882 * @brief Write a COMMON_COMMON_GET_PROFILE_CONFIG command. 2883 * 2884 * @param sli4 SLI context. 2885 * @param buf Destination buffer for the command. 2886 * @param size Buffer size in bytes. 2887 * @param dma DMA capable memory used to retrieve profile. 2888 * 2889 * @return Returns the number of bytes written. 2890 */ 2891 int32_t 2892 sli_cmd_common_get_profile_config(sli4_t *sli4, void *buf, size_t size, ocs_dma_t *dma) 2893 { 2894 sli4_req_common_get_profile_config_t *req = NULL; 2895 uint32_t sli_config_off = 0; 2896 uint32_t payload_size; 2897 2898 if (SLI4_PORT_TYPE_FC == sli4->port_type) { 2899 sli_config_off = sli_cmd_sli_config(sli4, buf, size, 2900 sizeof (sli4_req_common_get_profile_config_t), 2901 dma); 2902 } 2903 2904 if (dma != NULL) { 2905 req = dma->virt; 2906 ocs_memset(req, 0, dma->size); 2907 payload_size = dma->size; 2908 } else { 2909 req = (sli4_req_common_get_profile_config_t *)((uint8_t *)buf + sli_config_off); 2910 payload_size = sizeof(sli4_req_common_get_profile_config_t); 2911 } 2912 2913 req->hdr.opcode = SLI4_OPC_COMMON_GET_PROFILE_CONFIG; 2914 req->hdr.subsystem = SLI4_SUBSYSTEM_COMMON; 2915 req->hdr.request_length = payload_size - sizeof(sli4_req_hdr_t); 2916 req->hdr.version = 1; 2917 2918 return(sli_config_off + sizeof(sli4_req_common_get_profile_config_t)); 2919 } 2920 2921 /** 2922 * @ingroup sli 2923 * @brief Write a COMMON_COMMON_SET_PROFILE_CONFIG command. 2924 * 2925 * @param sli4 SLI context. 2926 * @param buf Destination buffer for the command. 2927 * @param size Buffer size, in bytes. 2928 * @param dma DMA capable memory containing profile. 2929 * @param profile_id Profile ID to configure. 2930 * @param descriptor_count Number of descriptors in DMA buffer. 2931 * @param isap Implicit Set Active Profile value to use. 2932 * 2933 * @return Returns the number of bytes written. 2934 */ 2935 int32_t 2936 sli_cmd_common_set_profile_config(sli4_t *sli4, void *buf, size_t size, ocs_dma_t *dma, 2937 uint8_t profile_id, uint32_t descriptor_count, uint8_t isap) 2938 { 2939 sli4_req_common_set_profile_config_t *req = NULL; 2940 uint32_t cmd_off = 0; 2941 uint32_t payload_size; 2942 2943 if (SLI4_PORT_TYPE_FC == sli4->port_type) { 2944 cmd_off = sli_cmd_sli_config(sli4, buf, size, 2945 sizeof (sli4_req_common_set_profile_config_t), 2946 dma); 2947 } 2948 2949 if (dma != NULL) { 2950 req = dma->virt; 2951 ocs_memset(req, 0, dma->size); 2952 payload_size = dma->size; 2953 } else { 2954 req = (sli4_req_common_set_profile_config_t *)((uint8_t *)buf + cmd_off); 2955 payload_size = sizeof(sli4_req_common_set_profile_config_t); 2956 } 2957 2958 req->hdr.opcode = SLI4_OPC_COMMON_SET_PROFILE_CONFIG; 2959 req->hdr.subsystem = SLI4_SUBSYSTEM_COMMON; 2960 req->hdr.request_length = payload_size - sizeof(sli4_req_hdr_t); 2961 req->hdr.version = 1; 2962 req->profile_id = profile_id; 2963 req->desc_count = descriptor_count; 2964 req->isap = isap; 2965 2966 return(cmd_off + sizeof(sli4_req_common_set_profile_config_t)); 2967 } 2968 2969 /** 2970 * @ingroup sli 2971 * @brief Write a COMMON_COMMON_GET_PROFILE_LIST command. 2972 * 2973 * @param sli4 SLI context. 2974 * @param buf Destination buffer for the command. 2975 * @param size Buffer size in bytes. 2976 * @param start_profile_index First profile index to return. 2977 * @param dma Buffer into which the list will be written. 2978 * 2979 * @return Returns the number of bytes written. 2980 */ 2981 int32_t 2982 sli_cmd_common_get_profile_list(sli4_t *sli4, void *buf, size_t size, 2983 uint32_t start_profile_index, ocs_dma_t *dma) 2984 { 2985 sli4_req_common_get_profile_list_t *req = NULL; 2986 uint32_t cmd_off = 0; 2987 uint32_t payload_size; 2988 2989 if (SLI4_PORT_TYPE_FC == sli4->port_type) { 2990 cmd_off = sli_cmd_sli_config(sli4, buf, size, 2991 sizeof (sli4_req_common_get_profile_list_t), 2992 dma); 2993 } 2994 2995 if (dma != NULL) { 2996 req = dma->virt; 2997 ocs_memset(req, 0, dma->size); 2998 payload_size = dma->size; 2999 } else { 3000 req = (sli4_req_common_get_profile_list_t *)((uint8_t *)buf + cmd_off); 3001 payload_size = sizeof(sli4_req_common_get_profile_list_t); 3002 } 3003 3004 req->hdr.opcode = SLI4_OPC_COMMON_GET_PROFILE_LIST; 3005 req->hdr.subsystem = SLI4_SUBSYSTEM_COMMON; 3006 req->hdr.request_length = payload_size - sizeof(sli4_req_hdr_t); 3007 req->hdr.version = 0; 3008 3009 req->start_profile_index = start_profile_index; 3010 3011 return(cmd_off + sizeof(sli4_req_common_get_profile_list_t)); 3012 } 3013 3014 /** 3015 * @ingroup sli 3016 * @brief Write a COMMON_COMMON_GET_ACTIVE_PROFILE command. 3017 * 3018 * @param sli4 SLI context. 3019 * @param buf Destination buffer for the command. 3020 * @param size Buffer size in bytes. 3021 * 3022 * @return Returns the number of bytes written. 3023 */ 3024 int32_t 3025 sli_cmd_common_get_active_profile(sli4_t *sli4, void *buf, size_t size) 3026 { 3027 sli4_req_common_get_active_profile_t *req = NULL; 3028 uint32_t cmd_off = 0; 3029 uint32_t payload_size; 3030 3031 /* Payload length must accommodate both request and response */ 3032 payload_size = max(sizeof(sli4_req_common_get_active_profile_t), 3033 sizeof(sli4_res_common_get_active_profile_t)); 3034 3035 if (SLI4_PORT_TYPE_FC == sli4->port_type) { 3036 cmd_off = sli_cmd_sli_config(sli4, buf, size, 3037 payload_size, 3038 NULL); 3039 } 3040 3041 req = (sli4_req_common_get_active_profile_t *) 3042 ((uint8_t*)buf + cmd_off); 3043 3044 req->hdr.opcode = SLI4_OPC_COMMON_GET_ACTIVE_PROFILE; 3045 req->hdr.subsystem = SLI4_SUBSYSTEM_COMMON; 3046 req->hdr.request_length = payload_size - sizeof(sli4_req_hdr_t); 3047 req->hdr.version = 0; 3048 3049 return(cmd_off + sizeof(sli4_req_common_get_active_profile_t)); 3050 } 3051 3052 /** 3053 * @ingroup sli 3054 * @brief Write a COMMON_COMMON_SET_ACTIVE_PROFILE command. 3055 * 3056 * @param sli4 SLI context. 3057 * @param buf Destination buffer for the command. 3058 * @param size Buffer size in bytes. 3059 * @param fd If non-zero, set profile to factory default. 3060 * @param active_profile_id ID of new active profile. 3061 * 3062 * @return Returns the number of bytes written. 3063 */ 3064 int32_t 3065 sli_cmd_common_set_active_profile(sli4_t *sli4, void *buf, size_t size, 3066 uint32_t fd, uint32_t active_profile_id) 3067 { 3068 sli4_req_common_set_active_profile_t *req = NULL; 3069 uint32_t cmd_off = 0; 3070 uint32_t payload_size; 3071 3072 /* Payload length must accommodate both request and response */ 3073 payload_size = max(sizeof(sli4_req_common_set_active_profile_t), 3074 sizeof(sli4_res_common_set_active_profile_t)); 3075 3076 if (SLI4_PORT_TYPE_FC == sli4->port_type) { 3077 cmd_off = sli_cmd_sli_config(sli4, buf, size, 3078 payload_size, 3079 NULL); 3080 } 3081 3082 req = (sli4_req_common_set_active_profile_t *) 3083 ((uint8_t*)buf + cmd_off); 3084 3085 req->hdr.opcode = SLI4_OPC_COMMON_SET_ACTIVE_PROFILE; 3086 req->hdr.subsystem = SLI4_SUBSYSTEM_COMMON; 3087 req->hdr.request_length = payload_size - sizeof(sli4_req_hdr_t); 3088 req->hdr.version = 0; 3089 req->fd = fd; 3090 req->active_profile_id = active_profile_id; 3091 3092 return(cmd_off + sizeof(sli4_req_common_set_active_profile_t)); 3093 } 3094 3095 /** 3096 * @ingroup sli 3097 * @brief Write a COMMON_GET_RECONFIG_LINK_INFO command. 3098 * 3099 * @param sli4 SLI context. 3100 * @param buf Destination buffer for the command. 3101 * @param size Buffer size in bytes. 3102 * @param dma Buffer to store the supported link configuration modes from the physical device. 3103 * 3104 * @return Returns the number of bytes written. 3105 */ 3106 int32_t 3107 sli_cmd_common_get_reconfig_link_info(sli4_t *sli4, void *buf, size_t size, ocs_dma_t *dma) 3108 { 3109 sli4_req_common_get_reconfig_link_info_t *req = NULL; 3110 uint32_t cmd_off = 0; 3111 uint32_t payload_size; 3112 3113 /* Payload length must accommodate both request and response */ 3114 payload_size = max(sizeof(sli4_req_common_get_reconfig_link_info_t), 3115 sizeof(sli4_res_common_get_reconfig_link_info_t)); 3116 3117 if (SLI4_PORT_TYPE_FC == sli4->port_type) { 3118 cmd_off = sli_cmd_sli_config(sli4, buf, size, 3119 payload_size, 3120 dma); 3121 } 3122 3123 if (dma != NULL) { 3124 req = dma->virt; 3125 ocs_memset(req, 0, dma->size); 3126 payload_size = dma->size; 3127 } else { 3128 req = (sli4_req_common_get_reconfig_link_info_t *)((uint8_t *)buf + cmd_off); 3129 payload_size = sizeof(sli4_req_common_get_reconfig_link_info_t); 3130 } 3131 3132 req->hdr.opcode = SLI4_OPC_COMMON_GET_RECONFIG_LINK_INFO; 3133 req->hdr.subsystem = SLI4_SUBSYSTEM_COMMON; 3134 req->hdr.request_length = payload_size - sizeof(sli4_req_hdr_t); 3135 req->hdr.version = 0; 3136 3137 return(cmd_off + sizeof(sli4_req_common_get_reconfig_link_info_t)); 3138 } 3139 3140 /** 3141 * @ingroup sli 3142 * @brief Write a COMMON_SET_RECONFIG_LINK_ID command. 3143 * 3144 * @param sli4 SLI context. 3145 * @param buf destination buffer for the command. 3146 * @param size buffer size in bytes. 3147 * @param fd If non-zero, set link config to factory default. 3148 * @param active_link_config_id ID of new active profile. 3149 * @param dma Buffer to assign the link configuration mode that is to become active from the physical device. 3150 * 3151 * @return Returns the number of bytes written. 3152 */ 3153 int32_t 3154 sli_cmd_common_set_reconfig_link_id(sli4_t *sli4, void *buf, size_t size, ocs_dma_t *dma, 3155 uint32_t fd, uint32_t active_link_config_id) 3156 { 3157 sli4_req_common_set_reconfig_link_id_t *req = NULL; 3158 uint32_t cmd_off = 0; 3159 uint32_t payload_size; 3160 3161 /* Payload length must accommodate both request and response */ 3162 payload_size = max(sizeof(sli4_req_common_set_reconfig_link_id_t), 3163 sizeof(sli4_res_common_set_reconfig_link_id_t)); 3164 3165 if (SLI4_PORT_TYPE_FC == sli4->port_type) { 3166 cmd_off = sli_cmd_sli_config(sli4, buf, size, 3167 payload_size, 3168 NULL); 3169 } 3170 3171 if (dma != NULL) { 3172 req = dma->virt; 3173 ocs_memset(req, 0, dma->size); 3174 payload_size = dma->size; 3175 } else { 3176 req = (sli4_req_common_set_reconfig_link_id_t *)((uint8_t *)buf + cmd_off); 3177 payload_size = sizeof(sli4_req_common_set_reconfig_link_id_t); 3178 } 3179 3180 req->hdr.opcode = SLI4_OPC_COMMON_SET_RECONFIG_LINK_ID; 3181 req->hdr.subsystem = SLI4_SUBSYSTEM_COMMON; 3182 req->hdr.request_length = payload_size - sizeof(sli4_req_hdr_t); 3183 req->hdr.version = 0; 3184 req->fd = fd; 3185 req->next_link_config_id = active_link_config_id; 3186 3187 return(cmd_off + sizeof(sli4_req_common_set_reconfig_link_id_t)); 3188 } 3189 3190 /** 3191 * @ingroup sli 3192 * @brief Check the mailbox/queue completion entry. 3193 * 3194 * @param buf Pointer to the MCQE. 3195 * 3196 * @return Returns 0 on success, or a non-zero value on failure. 3197 */ 3198 int32_t 3199 sli_cqe_mq(void *buf) 3200 { 3201 sli4_mcqe_t *mcqe = buf; 3202 3203 /* 3204 * Firmware can split mbx completions into two MCQEs: first with only 3205 * the "consumed" bit set and a second with the "complete" bit set. 3206 * Thus, ignore MCQE unless "complete" is set. 3207 */ 3208 if (!mcqe->cmp) { 3209 return -2; 3210 } 3211 3212 if (mcqe->completion_status) { 3213 ocs_log_debug(NULL, "bad status (cmpl=%#x ext=%#x con=%d cmp=%d ae=%d val=%d)\n", 3214 mcqe->completion_status, 3215 mcqe->extended_status, 3216 mcqe->con, 3217 mcqe->cmp, 3218 mcqe->ae, 3219 mcqe->val); 3220 } 3221 3222 return mcqe->completion_status; 3223 } 3224 3225 /** 3226 * @ingroup sli 3227 * @brief Check the asynchronous event completion entry. 3228 * 3229 * @param sli4 SLI context. 3230 * @param buf Pointer to the ACQE. 3231 * 3232 * @return Returns 0 on success, or a non-zero value on failure. 3233 */ 3234 int32_t 3235 sli_cqe_async(sli4_t *sli4, void *buf) 3236 { 3237 sli4_acqe_t *acqe = buf; 3238 int32_t rc = -1; 3239 3240 if (!sli4 || !buf) { 3241 ocs_log_err(NULL, "bad parameter sli4=%p buf=%p\n", sli4, buf); 3242 return -1; 3243 } 3244 3245 switch (acqe->event_code) { 3246 case SLI4_ACQE_EVENT_CODE_LINK_STATE: 3247 rc = sli_fc_process_link_state(sli4, buf); 3248 break; 3249 case SLI4_ACQE_EVENT_CODE_FCOE_FIP: 3250 rc = sli_fc_process_fcoe(sli4, buf); 3251 break; 3252 case SLI4_ACQE_EVENT_CODE_GRP_5: 3253 /*TODO*/ocs_log_debug(sli4->os, "ACQE GRP5\n"); 3254 break; 3255 case SLI4_ACQE_EVENT_CODE_SLI_PORT_EVENT: 3256 ocs_log_debug(sli4->os,"ACQE SLI Port, type=0x%x, data1,2=0x%08x,0x%08x\n", 3257 acqe->event_type, acqe->event_data[0], acqe->event_data[1]); 3258 #if defined(OCS_INCLUDE_DEBUG) 3259 ocs_dump32(OCS_DEBUG_ALWAYS, sli4->os, "acq", acqe, sizeof(*acqe)); 3260 #endif 3261 break; 3262 case SLI4_ACQE_EVENT_CODE_FC_LINK_EVENT: 3263 rc = sli_fc_process_link_attention(sli4, buf); 3264 break; 3265 default: 3266 /*TODO*/ocs_log_test(sli4->os, "ACQE unknown=%#x\n", acqe->event_code); 3267 } 3268 3269 return rc; 3270 } 3271 3272 /** 3273 * @brief Check the SLI_CONFIG response. 3274 * 3275 * @par Description 3276 * Function checks the SLI_CONFIG response and the payload status. 3277 * 3278 * @param buf Pointer to SLI_CONFIG response. 3279 * 3280 * @return Returns 0 on success, or a non-zero value on failure. 3281 */ 3282 static int32_t 3283 sli_res_sli_config(void *buf) 3284 { 3285 sli4_cmd_sli_config_t *sli_config = buf; 3286 3287 if (!buf || (SLI4_MBOX_COMMAND_SLI_CONFIG != sli_config->hdr.command)) { 3288 ocs_log_err(NULL, "bad parameter buf=%p cmd=%#x\n", buf, 3289 buf ? sli_config->hdr.command : -1); 3290 return -1; 3291 } 3292 3293 if (sli_config->hdr.status) { 3294 return sli_config->hdr.status; 3295 } 3296 3297 if (sli_config->emb) { 3298 return sli_config->payload.embed[4]; 3299 } else { 3300 ocs_log_test(NULL, "external buffers not supported\n"); 3301 return -1; 3302 } 3303 } 3304 3305 /** 3306 * @brief Issue a COMMON_FUNCTION_RESET command. 3307 * 3308 * @param sli4 SLI context. 3309 * 3310 * @return Returns 0 on success, or a non-zero value on failure. 3311 */ 3312 static int32_t 3313 sli_common_function_reset(sli4_t *sli4) 3314 { 3315 3316 if (sli_cmd_common_function_reset(sli4, sli4->bmbx.virt, SLI4_BMBX_SIZE)) { 3317 if (sli_bmbx_command(sli4)) { 3318 ocs_log_crit(sli4->os, "bootstrap mailbox write fail (COM_FUNC_RESET)\n"); 3319 return -1; 3320 } 3321 if (sli_res_sli_config(sli4->bmbx.virt)) { 3322 ocs_log_err(sli4->os, "bad status COM_FUNC_RESET\n"); 3323 return -1; 3324 } 3325 } else { 3326 ocs_log_err(sli4->os, "bad COM_FUNC_RESET write\n"); 3327 return -1; 3328 } 3329 3330 return 0; 3331 } 3332 3333 /** 3334 * @brief check to see if the FW is ready. 3335 * 3336 * @par Description 3337 * Based on <i>SLI-4 Architecture Specification, Revision 4.x0-13 (2012).</i>. 3338 * 3339 * @param sli4 SLI context. 3340 * @param timeout_ms Time, in milliseconds, to wait for the port to be ready 3341 * before failing. 3342 * 3343 * @return Returns TRUE for ready, or FALSE otherwise. 3344 */ 3345 static int32_t 3346 sli_wait_for_fw_ready(sli4_t *sli4, uint32_t timeout_ms) 3347 { 3348 uint32_t iter = timeout_ms / (SLI4_INIT_PORT_DELAY_US / 1000); 3349 uint32_t ready = FALSE; 3350 3351 do { 3352 iter--; 3353 ocs_udelay(SLI4_INIT_PORT_DELAY_US); 3354 if (sli_fw_ready(sli4) == 1) { 3355 ready = TRUE; 3356 } 3357 } while (!ready && (iter > 0)); 3358 3359 return ready; 3360 } 3361 3362 /** 3363 * @brief Initialize the firmware. 3364 * 3365 * @par Description 3366 * Based on <i>SLI-4 Architecture Specification, Revision 4.x0-13 (2012).</i>. 3367 * 3368 * @param sli4 SLI context. 3369 * 3370 * @return Returns 0 on success, or a non-zero value on failure. 3371 */ 3372 static int32_t 3373 sli_fw_init(sli4_t *sli4) 3374 { 3375 uint32_t ready; 3376 uint32_t endian; 3377 3378 /* 3379 * Is firmware ready for operation? 3380 */ 3381 ready = sli_wait_for_fw_ready(sli4, SLI4_FW_READY_TIMEOUT_MSEC); 3382 if (!ready) { 3383 ocs_log_crit(sli4->os, "FW status is NOT ready\n"); 3384 return -1; 3385 } 3386 3387 /* 3388 * Reset port to a known state 3389 */ 3390 switch (sli4->if_type) { 3391 case SLI4_IF_TYPE_BE3_SKH_PF: 3392 case SLI4_IF_TYPE_BE3_SKH_VF: 3393 /* No SLIPORT_CONTROL register so use command sequence instead */ 3394 if (sli_bmbx_wait(sli4, SLI4_BMBX_DELAY_US)) { 3395 ocs_log_crit(sli4->os, "bootstrap mailbox not ready\n"); 3396 return -1; 3397 } 3398 3399 if (sli_cmd_fw_initialize(sli4, sli4->bmbx.virt, SLI4_BMBX_SIZE)) { 3400 if (sli_bmbx_command(sli4)) { 3401 ocs_log_crit(sli4->os, "bootstrap mailbox write fail (FW_INIT)\n"); 3402 return -1; 3403 } 3404 } else { 3405 ocs_log_crit(sli4->os, "bad FW_INIT write\n"); 3406 return -1; 3407 } 3408 3409 if (sli_common_function_reset(sli4)) { 3410 ocs_log_err(sli4->os, "bad COM_FUNC_RESET write\n"); 3411 return -1; 3412 } 3413 break; 3414 case SLI4_IF_TYPE_LANCER_FC_ETH: 3415 case SLI4_IF_TYPE_LANCER_G7: 3416 #if BYTE_ORDER == LITTLE_ENDIAN 3417 endian = SLI4_SLIPORT_CONTROL_LITTLE_ENDIAN; 3418 #else 3419 endian = SLI4_SLIPORT_CONTROL_BIG_ENDIAN; 3420 #endif 3421 3422 if (sli_sliport_control(sli4, endian)) 3423 return -1; 3424 break; 3425 default: 3426 ocs_log_test(sli4->os, "if_type %d not supported\n", sli4->if_type); 3427 return -1; 3428 } 3429 3430 return 0; 3431 } 3432 3433 /** 3434 * @brief Terminate the firmware. 3435 * 3436 * @param sli4 SLI context. 3437 * 3438 * @return Returns 0 on success, or a non-zero value on failure. 3439 */ 3440 static int32_t 3441 sli_fw_term(sli4_t *sli4) 3442 { 3443 uint32_t endian; 3444 3445 if (sli4->if_type == SLI4_IF_TYPE_BE3_SKH_PF || 3446 sli4->if_type == SLI4_IF_TYPE_BE3_SKH_VF) { 3447 /* No SLIPORT_CONTROL register so use command sequence instead */ 3448 if (sli_bmbx_wait(sli4, SLI4_BMBX_DELAY_US)) { 3449 ocs_log_crit(sli4->os, "bootstrap mailbox not ready\n"); 3450 return -1; 3451 } 3452 3453 if (sli_common_function_reset(sli4)) { 3454 ocs_log_err(sli4->os, "bad COM_FUNC_RESET write\n"); 3455 return -1; 3456 } 3457 3458 if (sli_cmd_fw_deinitialize(sli4, sli4->bmbx.virt, SLI4_BMBX_SIZE)) { 3459 if (sli_bmbx_command(sli4)) { 3460 ocs_log_crit(sli4->os, "bootstrap mailbox write fail (FW_DEINIT)\n"); 3461 return -1; 3462 } 3463 } else { 3464 ocs_log_test(sli4->os, "bad FW_DEINIT write\n"); 3465 return -1; 3466 } 3467 } else { 3468 #if BYTE_ORDER == LITTLE_ENDIAN 3469 endian = SLI4_SLIPORT_CONTROL_LITTLE_ENDIAN; 3470 #else 3471 endian = SLI4_SLIPORT_CONTROL_BIG_ENDIAN; 3472 #endif 3473 /* type 2 etc. use SLIPORT_CONTROL to initialize port */ 3474 sli_sliport_control(sli4, endian); 3475 } 3476 return 0; 3477 } 3478 3479 /** 3480 * @brief Write the doorbell register associated with the queue object. 3481 * 3482 * @param sli4 SLI context. 3483 * @param q Queue object. 3484 * 3485 * @return Returns 0 on success, or a non-zero value on failure. 3486 */ 3487 static int32_t 3488 sli_queue_doorbell(sli4_t *sli4, sli4_queue_t *q) 3489 { 3490 uint32_t val = 0; 3491 3492 switch (q->type) { 3493 case SLI_QTYPE_EQ: 3494 if (sli4->if_type == SLI4_IF_TYPE_LANCER_G7) 3495 val = sli_iftype6_eq_doorbell(q->n_posted, q->id, FALSE); 3496 else 3497 val = sli_eq_doorbell(q->n_posted, q->id, FALSE); 3498 ocs_reg_write32(sli4->os, q->doorbell_rset, q->doorbell_offset, val); 3499 break; 3500 case SLI_QTYPE_CQ: 3501 if (sli4->if_type == SLI4_IF_TYPE_LANCER_G7) 3502 val = sli_iftype6_cq_doorbell(q->n_posted, q->id, FALSE); 3503 else 3504 val = sli_cq_doorbell(q->n_posted, q->id, FALSE); 3505 ocs_reg_write32(sli4->os, q->doorbell_rset, q->doorbell_offset, val); 3506 break; 3507 case SLI_QTYPE_MQ: 3508 val = SLI4_MQ_DOORBELL(q->n_posted, q->id); 3509 ocs_reg_write32(sli4->os, q->doorbell_rset, q->doorbell_offset, val); 3510 break; 3511 case SLI_QTYPE_RQ: 3512 { 3513 uint32_t n_posted = q->n_posted; 3514 /* 3515 * FC/FCoE has different rules for Receive Queues. The host 3516 * should only update the doorbell of the RQ-pair containing 3517 * the headers since the header / payload RQs are treated 3518 * as a matched unit. 3519 */ 3520 if (SLI4_PORT_TYPE_FC == sli4->port_type) { 3521 /* 3522 * In RQ-pair, an RQ either contains the FC header 3523 * (i.e. is_hdr == TRUE) or the payload. 3524 * 3525 * Don't ring doorbell for payload RQ 3526 */ 3527 if (!q->u.flag.is_hdr) { 3528 break; 3529 } 3530 /* 3531 * Some RQ cannot be incremented one entry at a time. Instead, 3532 * the driver collects a number of entries and updates the 3533 * RQ in batches. 3534 */ 3535 if (q->u.flag.rq_batch) { 3536 if (((q->index + q->n_posted) % SLI4_QUEUE_RQ_BATCH)) { 3537 break; 3538 } 3539 n_posted = SLI4_QUEUE_RQ_BATCH; 3540 } 3541 } 3542 3543 val = SLI4_RQ_DOORBELL(n_posted, q->id); 3544 ocs_reg_write32(sli4->os, q->doorbell_rset, q->doorbell_offset, val); 3545 break; 3546 } 3547 case SLI_QTYPE_WQ: 3548 if (sli4->if_type == SLI4_IF_TYPE_LANCER_G7) { 3549 val = SLI4_WQ_DOORBELL(q->n_posted, 0, q->id); 3550 } else { 3551 /* For iftype = 2 and 3, q->index value is ignored */ 3552 val = SLI4_WQ_DOORBELL(q->n_posted, q->index, q->id); 3553 } 3554 3555 ocs_reg_write32(sli4->os, q->doorbell_rset, q->doorbell_offset, val); 3556 break; 3557 default: 3558 ocs_log_test(sli4->os, "bad queue type %d\n", q->type); 3559 return -1; 3560 } 3561 3562 return 0; 3563 } 3564 3565 static int32_t 3566 sli_request_features(sli4_t *sli4, sli4_features_t *features, uint8_t query) 3567 { 3568 3569 if (sli_cmd_request_features(sli4, sli4->bmbx.virt, SLI4_BMBX_SIZE, 3570 *features, query)) { 3571 sli4_cmd_request_features_t *req_features = sli4->bmbx.virt; 3572 3573 if (sli_bmbx_command(sli4)) { 3574 ocs_log_crit(sli4->os, "bootstrap mailbox write fail (REQUEST_FEATURES)\n"); 3575 return -1; 3576 } 3577 if (req_features->hdr.status) { 3578 ocs_log_err(sli4->os, "REQUEST_FEATURES bad status %#x\n", 3579 req_features->hdr.status); 3580 return -1; 3581 } 3582 features->dword = req_features->response.dword; 3583 } else { 3584 ocs_log_err(sli4->os, "bad REQUEST_FEATURES write\n"); 3585 return -1; 3586 } 3587 3588 return 0; 3589 } 3590 3591 /** 3592 * @brief Calculate max queue entries. 3593 * 3594 * @param sli4 SLI context. 3595 * 3596 * @return Returns 0 on success, or a non-zero value on failure. 3597 */ 3598 void 3599 sli_calc_max_qentries(sli4_t *sli4) 3600 { 3601 sli4_qtype_e q; 3602 uint32_t alloc_size, qentries, qentry_size; 3603 3604 for (q = SLI_QTYPE_EQ; q < SLI_QTYPE_MAX; q++) { 3605 sli4->config.max_qentries[q] = sli_convert_mask_to_count(sli4->config.count_method[q], 3606 sli4->config.count_mask[q]); 3607 } 3608 3609 /* single, continguous DMA allocations will be called for each queue 3610 * of size (max_qentries * queue entry size); since these can be large, 3611 * check against the OS max DMA allocation size 3612 */ 3613 for (q = SLI_QTYPE_EQ; q < SLI_QTYPE_MAX; q++) { 3614 qentries = sli4->config.max_qentries[q]; 3615 qentry_size = sli_get_queue_entry_size(sli4, q); 3616 alloc_size = qentries * qentry_size; 3617 if (alloc_size > ocs_max_dma_alloc(sli4->os, SLI_PAGE_SIZE)) { 3618 while (alloc_size > ocs_max_dma_alloc(sli4->os, SLI_PAGE_SIZE)) { 3619 /* cut the qentries in hwf until alloc_size <= max DMA alloc size */ 3620 qentries >>= 1; 3621 alloc_size = qentries * qentry_size; 3622 } 3623 ocs_log_debug(sli4->os, "[%s]: max_qentries from %d to %d (max dma %d)\n", 3624 SLI_QNAME[q], sli4->config.max_qentries[q], 3625 qentries, ocs_max_dma_alloc(sli4->os, SLI_PAGE_SIZE)); 3626 sli4->config.max_qentries[q] = qentries; 3627 } 3628 } 3629 } 3630 3631 /** 3632 * @brief Issue a FW_CONFIG mailbox command and store the results. 3633 * 3634 * @param sli4 SLI context. 3635 * 3636 * @return Returns 0 on success, or a non-zero value on failure. 3637 */ 3638 static int32_t 3639 sli_query_fw_config(sli4_t *sli4) 3640 { 3641 /* 3642 * Read the device configuration 3643 * 3644 * Note: Only ulp0 fields contain values 3645 */ 3646 if (sli_cmd_common_query_fw_config(sli4, sli4->bmbx.virt, SLI4_BMBX_SIZE)) { 3647 sli4_res_common_query_fw_config_t *fw_config = 3648 (sli4_res_common_query_fw_config_t *) 3649 (((uint8_t *)sli4->bmbx.virt) + offsetof(sli4_cmd_sli_config_t, payload.embed)); 3650 3651 if (sli_bmbx_command(sli4)) { 3652 ocs_log_crit(sli4->os, "bootstrap mailbox write fail (QUERY_FW_CONFIG)\n"); 3653 return -1; 3654 } 3655 if (fw_config->hdr.status) { 3656 ocs_log_err(sli4->os, "COMMON_QUERY_FW_CONFIG bad status %#x\n", 3657 fw_config->hdr.status); 3658 return -1; 3659 } 3660 3661 sli4->physical_port = fw_config->physical_port; 3662 sli4->config.dual_ulp_capable = ((fw_config->function_mode & SLI4_FUNCTION_MODE_DUA_MODE) == 0 ? 0 : 1); 3663 sli4->config.is_ulp_fc[0] = ((fw_config->ulp0_mode & 3664 (SLI4_ULP_MODE_FCOE_INI | 3665 SLI4_ULP_MODE_FCOE_TGT)) == 0 ? 0 : 1); 3666 sli4->config.is_ulp_fc[1] = ((fw_config->ulp1_mode & 3667 (SLI4_ULP_MODE_FCOE_INI | 3668 SLI4_ULP_MODE_FCOE_TGT)) == 0 ? 0 : 1); 3669 3670 if (sli4->config.dual_ulp_capable) { 3671 /* 3672 * Lancer will not support this, so we use the values 3673 * from the READ_CONFIG. 3674 */ 3675 if (sli4->config.is_ulp_fc[0] && 3676 sli4->config.is_ulp_fc[1]) { 3677 sli4->config.max_qcount[SLI_QTYPE_WQ] = fw_config->ulp0_toe_wq_total + fw_config->ulp1_toe_wq_total; 3678 sli4->config.max_qcount[SLI_QTYPE_RQ] = fw_config->ulp0_toe_defrq_total + fw_config->ulp1_toe_defrq_total; 3679 } else if (sli4->config.is_ulp_fc[0]) { 3680 sli4->config.max_qcount[SLI_QTYPE_WQ] = fw_config->ulp0_toe_wq_total; 3681 sli4->config.max_qcount[SLI_QTYPE_RQ] = fw_config->ulp0_toe_defrq_total; 3682 } else { 3683 sli4->config.max_qcount[SLI_QTYPE_WQ] = fw_config->ulp1_toe_wq_total; 3684 sli4->config.max_qcount[SLI_QTYPE_RQ] = fw_config->ulp1_toe_defrq_total; 3685 } 3686 } 3687 } else { 3688 ocs_log_err(sli4->os, "bad QUERY_FW_CONFIG write\n"); 3689 return -1; 3690 } 3691 return 0; 3692 } 3693 3694 static int32_t 3695 sli_get_config(sli4_t *sli4) 3696 { 3697 ocs_dma_t get_cntl_addl_data; 3698 3699 /* 3700 * Read the device configuration 3701 */ 3702 if (sli_cmd_read_config(sli4, sli4->bmbx.virt, SLI4_BMBX_SIZE)) { 3703 sli4_res_read_config_t *read_config = sli4->bmbx.virt; 3704 uint32_t i; 3705 uint32_t total; 3706 3707 if (sli_bmbx_command(sli4)) { 3708 ocs_log_crit(sli4->os, "bootstrap mailbox write fail (READ_CONFIG)\n"); 3709 return -1; 3710 } 3711 if (read_config->hdr.status) { 3712 ocs_log_err(sli4->os, "READ_CONFIG bad status %#x\n", 3713 read_config->hdr.status); 3714 return -1; 3715 } 3716 3717 sli4->config.has_extents = read_config->ext; 3718 if (FALSE == sli4->config.has_extents) { 3719 uint32_t i = 0; 3720 uint32_t *base = sli4->config.extent[0].base; 3721 3722 if (!base) { 3723 if (NULL == (base = ocs_malloc(sli4->os, SLI_RSRC_MAX * sizeof(uint32_t), 3724 OCS_M_ZERO | OCS_M_NOWAIT))) { 3725 ocs_log_err(sli4->os, "memory allocation failed for sli4_resource_t\n"); 3726 return -1; 3727 } 3728 } 3729 3730 for (i = 0; i < SLI_RSRC_MAX; i++) { 3731 sli4->config.extent[i].number = 1; 3732 sli4->config.extent[i].n_alloc = 0; 3733 sli4->config.extent[i].base = &base[i]; 3734 } 3735 3736 sli4->config.extent[SLI_RSRC_FCOE_VFI].base[0] = read_config->vfi_base; 3737 sli4->config.extent[SLI_RSRC_FCOE_VFI].size = read_config->vfi_count; 3738 3739 sli4->config.extent[SLI_RSRC_FCOE_VPI].base[0] = read_config->vpi_base; 3740 sli4->config.extent[SLI_RSRC_FCOE_VPI].size = read_config->vpi_count; 3741 3742 sli4->config.extent[SLI_RSRC_FCOE_RPI].base[0] = read_config->rpi_base; 3743 sli4->config.extent[SLI_RSRC_FCOE_RPI].size = read_config->rpi_count; 3744 3745 sli4->config.extent[SLI_RSRC_FCOE_XRI].base[0] = read_config->xri_base; 3746 sli4->config.extent[SLI_RSRC_FCOE_XRI].size = OCS_MIN(255,read_config->xri_count); 3747 3748 sli4->config.extent[SLI_RSRC_FCOE_FCFI].base[0] = 0; 3749 sli4->config.extent[SLI_RSRC_FCOE_FCFI].size = read_config->fcfi_count; 3750 } else { 3751 /* TODO extents*/ 3752 ; 3753 } 3754 3755 for (i = 0; i < SLI_RSRC_MAX; i++) { 3756 total = sli4->config.extent[i].number * sli4->config.extent[i].size; 3757 sli4->config.extent[i].use_map = ocs_bitmap_alloc(total); 3758 if (NULL == sli4->config.extent[i].use_map) { 3759 ocs_log_err(sli4->os, "bitmap memory allocation failed " 3760 "resource %d\n", i); 3761 return -1; 3762 } 3763 sli4->config.extent[i].map_size = total; 3764 } 3765 3766 sli4->config.topology = read_config->topology; 3767 sli4->config.ptv = read_config->ptv; 3768 if (sli4->config.ptv){ 3769 sli4->config.tf = read_config->tf; 3770 sli4->config.pt = read_config->pt; 3771 } 3772 ocs_log_info(sli4->os, "Persistent Topology: PTV: %d, TF: %d, PT: %d \n", 3773 sli4->config.topology, sli4->config.ptv, sli4->config.tf, sli4->config.pt); 3774 3775 switch (sli4->config.topology) { 3776 case SLI4_READ_CFG_TOPO_FCOE: 3777 ocs_log_debug(sli4->os, "FCoE\n"); 3778 break; 3779 case SLI4_READ_CFG_TOPO_FC: 3780 ocs_log_debug(sli4->os, "FC (unknown)\n"); 3781 break; 3782 case SLI4_READ_CFG_TOPO_FC_DA: 3783 ocs_log_debug(sli4->os, "FC (direct attach)\n"); 3784 break; 3785 case SLI4_READ_CFG_TOPO_FC_AL: 3786 ocs_log_debug(sli4->os, "FC (arbitrated loop)\n"); 3787 break; 3788 default: 3789 ocs_log_test(sli4->os, "bad topology %#x\n", sli4->config.topology); 3790 } 3791 3792 sli4->config.e_d_tov = read_config->e_d_tov; 3793 sli4->config.r_a_tov = read_config->r_a_tov; 3794 3795 sli4->config.link_module_type = read_config->lmt; 3796 3797 sli4->config.max_qcount[SLI_QTYPE_EQ] = read_config->eq_count; 3798 sli4->config.max_qcount[SLI_QTYPE_CQ] = read_config->cq_count; 3799 sli4->config.max_qcount[SLI_QTYPE_WQ] = read_config->wq_count; 3800 sli4->config.max_qcount[SLI_QTYPE_RQ] = read_config->rq_count; 3801 3802 /* 3803 * READ_CONFIG doesn't give the max number of MQ. Applications 3804 * will typically want 1, but we may need another at some future 3805 * date. Dummy up a "max" MQ count here. 3806 */ 3807 sli4->config.max_qcount[SLI_QTYPE_MQ] = SLI_USER_MQ_COUNT; 3808 } else { 3809 ocs_log_err(sli4->os, "bad READ_CONFIG write\n"); 3810 return -1; 3811 } 3812 3813 if (sli_cmd_common_get_sli4_parameters(sli4, sli4->bmbx.virt, SLI4_BMBX_SIZE)) { 3814 sli4_res_common_get_sli4_parameters_t *parms = (sli4_res_common_get_sli4_parameters_t *) 3815 (((uint8_t *)sli4->bmbx.virt) + offsetof(sli4_cmd_sli_config_t, payload.embed)); 3816 3817 if (sli_bmbx_command(sli4)) { 3818 ocs_log_crit(sli4->os, "bootstrap mailbox write fail (COMMON_GET_SLI4_PARAMETERS)\n"); 3819 return -1; 3820 } else if (parms->hdr.status) { 3821 ocs_log_err(sli4->os, "COMMON_GET_SLI4_PARAMETERS bad status %#x att'l %#x\n", 3822 parms->hdr.status, parms->hdr.additional_status); 3823 return -1; 3824 } 3825 3826 sli4->config.auto_reg = parms->areg; 3827 sli4->config.auto_xfer_rdy = parms->agxf; 3828 sli4->config.hdr_template_req = parms->hdrr; 3829 sli4->config.t10_dif_inline_capable = parms->timm; 3830 sli4->config.t10_dif_separate_capable = parms->tsmm; 3831 3832 sli4->config.mq_create_version = parms->mqv; 3833 sli4->config.cq_create_version = parms->cqv; 3834 sli4->config.rq_min_buf_size = parms->min_rq_buffer_size; 3835 sli4->config.rq_max_buf_size = parms->max_rq_buffer_size; 3836 3837 sli4->config.qpage_count[SLI_QTYPE_EQ] = parms->eq_page_cnt; 3838 sli4->config.qpage_count[SLI_QTYPE_CQ] = parms->cq_page_cnt; 3839 sli4->config.qpage_count[SLI_QTYPE_MQ] = parms->mq_page_cnt; 3840 sli4->config.qpage_count[SLI_QTYPE_WQ] = parms->wq_page_cnt; 3841 sli4->config.qpage_count[SLI_QTYPE_RQ] = parms->rq_page_cnt; 3842 3843 /* save count methods and masks for each queue type */ 3844 sli4->config.count_mask[SLI_QTYPE_EQ] = parms->eqe_count_mask; 3845 sli4->config.count_method[SLI_QTYPE_EQ] = parms->eqe_count_method; 3846 sli4->config.count_mask[SLI_QTYPE_CQ] = parms->cqe_count_mask; 3847 sli4->config.count_method[SLI_QTYPE_CQ] = parms->cqe_count_method; 3848 sli4->config.count_mask[SLI_QTYPE_MQ] = parms->mqe_count_mask; 3849 sli4->config.count_method[SLI_QTYPE_MQ] = parms->mqe_count_method; 3850 sli4->config.count_mask[SLI_QTYPE_WQ] = parms->wqe_count_mask; 3851 sli4->config.count_method[SLI_QTYPE_WQ] = parms->wqe_count_method; 3852 sli4->config.count_mask[SLI_QTYPE_RQ] = parms->rqe_count_mask; 3853 sli4->config.count_method[SLI_QTYPE_RQ] = parms->rqe_count_method; 3854 3855 /* now calculate max queue entries */ 3856 sli_calc_max_qentries(sli4); 3857 3858 sli4->config.max_sgl_pages = parms->sgl_page_cnt; /* max # of pages */ 3859 sli4->config.sgl_page_sizes = parms->sgl_page_sizes; /* bit map of available sizes */ 3860 /* ignore HLM here. Use value from REQUEST_FEATURES */ 3861 3862 sli4->config.sge_supported_length = parms->sge_supported_length; 3863 if (sli4->config.sge_supported_length > OCS_MAX_SGE_SIZE) 3864 sli4->config.sge_supported_length = OCS_MAX_SGE_SIZE; 3865 3866 sli4->config.sgl_pre_registration_required = parms->sglr; 3867 /* default to using pre-registered SGL's */ 3868 sli4->config.sgl_pre_registered = TRUE; 3869 3870 sli4->config.perf_hint = parms->phon; 3871 sli4->config.perf_wq_id_association = parms->phwq; 3872 3873 sli4->config.rq_batch = parms->rq_db_window; 3874 3875 /* save the fields for skyhawk SGL chaining */ 3876 sli4->config.sgl_chaining_params.chaining_capable = 3877 (parms->sglc == 1); 3878 sli4->config.sgl_chaining_params.frag_num_field_offset = 3879 parms->frag_num_field_offset; 3880 sli4->config.sgl_chaining_params.frag_num_field_mask = 3881 (1ull << parms->frag_num_field_size) - 1; 3882 sli4->config.sgl_chaining_params.sgl_index_field_offset = 3883 parms->sgl_index_field_offset; 3884 sli4->config.sgl_chaining_params.sgl_index_field_mask = 3885 (1ull << parms->sgl_index_field_size) - 1; 3886 sli4->config.sgl_chaining_params.chain_sge_initial_value_lo = 3887 parms->chain_sge_initial_value_lo; 3888 sli4->config.sgl_chaining_params.chain_sge_initial_value_hi = 3889 parms->chain_sge_initial_value_hi; 3890 3891 /* Use the highest available WQE size. */ 3892 if (parms->wqe_sizes & SLI4_128BYTE_WQE_SUPPORT) { 3893 sli4->config.wqe_size = SLI4_WQE_EXT_BYTES; 3894 } else { 3895 sli4->config.wqe_size = SLI4_WQE_BYTES; 3896 } 3897 } 3898 3899 if (sli_query_fw_config(sli4)) { 3900 ocs_log_err(sli4->os, "Error sending QUERY_FW_CONFIG\n"); 3901 return -1; 3902 } 3903 3904 sli4->config.port_number = 0; 3905 3906 /* 3907 * Issue COMMON_GET_CNTL_ATTRIBUTES to get port_number. Temporarily 3908 * uses VPD DMA buffer as the response won't fit in the embedded 3909 * buffer. 3910 */ 3911 if (sli_cmd_common_get_cntl_attributes(sli4, sli4->bmbx.virt, SLI4_BMBX_SIZE, &sli4->vpd.data)) { 3912 sli4_res_common_get_cntl_attributes_t *attr = sli4->vpd.data.virt; 3913 3914 if (sli_bmbx_command(sli4)) { 3915 ocs_log_crit(sli4->os, "bootstrap mailbox write fail (COMMON_GET_CNTL_ATTRIBUTES)\n"); 3916 return -1; 3917 } else if (attr->hdr.status) { 3918 ocs_log_err(sli4->os, "COMMON_GET_CNTL_ATTRIBUTES bad status %#x att'l %#x\n", 3919 attr->hdr.status, attr->hdr.additional_status); 3920 return -1; 3921 } 3922 3923 sli4->config.port_number = attr->port_number; 3924 3925 ocs_memcpy(sli4->config.bios_version_string, attr->bios_version_string, 3926 sizeof(sli4->config.bios_version_string)); 3927 } else { 3928 ocs_log_err(sli4->os, "bad COMMON_GET_CNTL_ATTRIBUTES write\n"); 3929 return -1; 3930 } 3931 3932 if (ocs_dma_alloc(sli4->os, &get_cntl_addl_data, sizeof(sli4_res_common_get_cntl_addl_attributes_t), 3933 OCS_MIN_DMA_ALIGNMENT)) { 3934 ocs_log_err(sli4->os, "Failed to allocate memory for GET_CNTL_ADDL_ATTR data\n"); 3935 } else { 3936 if (sli_cmd_common_get_cntl_addl_attributes(sli4, sli4->bmbx.virt, SLI4_BMBX_SIZE, 3937 &get_cntl_addl_data)) { 3938 sli4_res_common_get_cntl_addl_attributes_t *attr = get_cntl_addl_data.virt; 3939 3940 if (sli_bmbx_command(sli4)) { 3941 ocs_log_crit(sli4->os, 3942 "bootstrap mailbox write fail (COMMON_GET_CNTL_ADDL_ATTRIBUTES)\n"); 3943 ocs_dma_free(sli4->os, &get_cntl_addl_data); 3944 return -1; 3945 } 3946 if (attr->hdr.status) { 3947 ocs_log_err(sli4->os, "COMMON_GET_CNTL_ADDL_ATTRIBUTES bad status %#x\n", 3948 attr->hdr.status); 3949 ocs_dma_free(sli4->os, &get_cntl_addl_data); 3950 return -1; 3951 } 3952 3953 ocs_memcpy(sli4->config.ipl_name, attr->ipl_file_name, sizeof(sli4->config.ipl_name)); 3954 3955 ocs_log_debug(sli4->os, "IPL:%s \n", (char*)sli4->config.ipl_name); 3956 } else { 3957 ocs_log_err(sli4->os, "bad COMMON_GET_CNTL_ADDL_ATTRIBUTES write\n"); 3958 ocs_dma_free(sli4->os, &get_cntl_addl_data); 3959 return -1; 3960 } 3961 3962 ocs_dma_free(sli4->os, &get_cntl_addl_data); 3963 } 3964 3965 if (sli_cmd_common_get_port_name(sli4, sli4->bmbx.virt, SLI4_BMBX_SIZE)) { 3966 sli4_res_common_get_port_name_t *port_name = (sli4_res_common_get_port_name_t *)(((uint8_t *)sli4->bmbx.virt) + 3967 offsetof(sli4_cmd_sli_config_t, payload.embed)); 3968 3969 if (sli_bmbx_command(sli4)) { 3970 ocs_log_crit(sli4->os, "bootstrap mailbox write fail (COMMON_GET_PORT_NAME)\n"); 3971 return -1; 3972 } 3973 3974 sli4->config.port_name[0] = port_name->port_name[sli4->config.port_number]; 3975 } 3976 sli4->config.port_name[1] = '\0'; 3977 3978 if (sli_cmd_read_rev(sli4, sli4->bmbx.virt, SLI4_BMBX_SIZE, &sli4->vpd.data)) { 3979 sli4_cmd_read_rev_t *read_rev = sli4->bmbx.virt; 3980 3981 if (sli_bmbx_command(sli4)) { 3982 ocs_log_crit(sli4->os, "bootstrap mailbox write fail (READ_REV)\n"); 3983 return -1; 3984 } 3985 if (read_rev->hdr.status) { 3986 ocs_log_err(sli4->os, "READ_REV bad status %#x\n", 3987 read_rev->hdr.status); 3988 return -1; 3989 } 3990 3991 sli4->config.fw_rev[0] = read_rev->first_fw_id; 3992 ocs_memcpy(sli4->config.fw_name[0],read_rev->first_fw_name, sizeof(sli4->config.fw_name[0])); 3993 3994 sli4->config.fw_rev[1] = read_rev->second_fw_id; 3995 ocs_memcpy(sli4->config.fw_name[1],read_rev->second_fw_name, sizeof(sli4->config.fw_name[1])); 3996 3997 sli4->config.hw_rev[0] = read_rev->first_hw_revision; 3998 sli4->config.hw_rev[1] = read_rev->second_hw_revision; 3999 sli4->config.hw_rev[2] = read_rev->third_hw_revision; 4000 4001 ocs_log_debug(sli4->os, "FW1:%s (%08x) / FW2:%s (%08x)\n", 4002 read_rev->first_fw_name, read_rev->first_fw_id, 4003 read_rev->second_fw_name, read_rev->second_fw_id); 4004 4005 ocs_log_debug(sli4->os, "HW1: %08x / HW2: %08x\n", read_rev->first_hw_revision, 4006 read_rev->second_hw_revision); 4007 4008 /* Check that all VPD data was returned */ 4009 if (read_rev->returned_vpd_length != read_rev->actual_vpd_length) { 4010 ocs_log_test(sli4->os, "VPD length: available=%d returned=%d actual=%d\n", 4011 read_rev->available_length, 4012 read_rev->returned_vpd_length, 4013 read_rev->actual_vpd_length); 4014 } 4015 sli4->vpd.length = read_rev->returned_vpd_length; 4016 } else { 4017 ocs_log_err(sli4->os, "bad READ_REV write\n"); 4018 return -1; 4019 } 4020 4021 if (sli_cmd_read_nvparms(sli4, sli4->bmbx.virt, SLI4_BMBX_SIZE)) { 4022 sli4_cmd_read_nvparms_t *read_nvparms = sli4->bmbx.virt; 4023 4024 if (sli_bmbx_command(sli4)) { 4025 ocs_log_crit(sli4->os, "bootstrap mailbox write fail (READ_NVPARMS)\n"); 4026 return -1; 4027 } 4028 if (read_nvparms->hdr.status) { 4029 ocs_log_err(sli4->os, "READ_NVPARMS bad status %#x\n", 4030 read_nvparms->hdr.status); 4031 return -1; 4032 } 4033 4034 ocs_memcpy(sli4->config.wwpn, read_nvparms->wwpn, sizeof(sli4->config.wwpn)); 4035 ocs_memcpy(sli4->config.wwnn, read_nvparms->wwnn, sizeof(sli4->config.wwnn)); 4036 4037 ocs_log_debug(sli4->os, "WWPN %02x:%02x:%02x:%02x:%02x:%02x:%02x:%02x\n", 4038 sli4->config.wwpn[0], 4039 sli4->config.wwpn[1], 4040 sli4->config.wwpn[2], 4041 sli4->config.wwpn[3], 4042 sli4->config.wwpn[4], 4043 sli4->config.wwpn[5], 4044 sli4->config.wwpn[6], 4045 sli4->config.wwpn[7]); 4046 ocs_log_debug(sli4->os, "WWNN %02x:%02x:%02x:%02x:%02x:%02x:%02x:%02x\n", 4047 sli4->config.wwnn[0], 4048 sli4->config.wwnn[1], 4049 sli4->config.wwnn[2], 4050 sli4->config.wwnn[3], 4051 sli4->config.wwnn[4], 4052 sli4->config.wwnn[5], 4053 sli4->config.wwnn[6], 4054 sli4->config.wwnn[7]); 4055 } else { 4056 ocs_log_err(sli4->os, "bad READ_NVPARMS write\n"); 4057 return -1; 4058 } 4059 4060 return 0; 4061 } 4062 4063 /**************************************************************************** 4064 * Public functions 4065 */ 4066 4067 /** 4068 * @ingroup sli 4069 * @brief Set up the SLI context. 4070 * 4071 * @param sli4 SLI context. 4072 * @param os Device abstraction. 4073 * @param port_type Protocol type of port (for example, FC and NIC). 4074 * 4075 * @return Returns 0 on success, or a non-zero value on failure. 4076 */ 4077 int32_t 4078 sli_setup(sli4_t *sli4, ocs_os_handle_t os, sli4_port_type_e port_type) 4079 { 4080 uint32_t sli_intf = UINT32_MAX; 4081 uint32_t pci_class_rev = 0; 4082 uint32_t rev_id = 0; 4083 uint32_t family = 0; 4084 uint32_t i; 4085 sli4_asic_entry_t *asic; 4086 4087 ocs_memset(sli4, 0, sizeof(sli4_t)); 4088 4089 sli4->os = os; 4090 sli4->port_type = port_type; 4091 4092 /* 4093 * Read the SLI_INTF register to discover the register layout 4094 * and other capability information 4095 */ 4096 sli_intf = ocs_config_read32(os, SLI4_INTF_REG); 4097 4098 if (sli_intf_valid_check(sli_intf)) { 4099 ocs_log_err(os, "SLI_INTF is not valid\n"); 4100 return -1; 4101 } 4102 4103 /* driver only support SLI-4 */ 4104 sli4->sli_rev = sli_intf_sli_revision(sli_intf); 4105 if (4 != sli4->sli_rev) { 4106 ocs_log_err(os, "Unsupported SLI revision (intf=%#x)\n", 4107 sli_intf); 4108 return -1; 4109 } 4110 4111 sli4->sli_family = sli_intf_sli_family(sli_intf); 4112 4113 sli4->if_type = sli_intf_if_type(sli_intf); 4114 4115 if ((SLI4_IF_TYPE_LANCER_FC_ETH == sli4->if_type) || 4116 (SLI4_IF_TYPE_LANCER_G7 == sli4->if_type)) { 4117 ocs_log_debug(os, "status=%#x error1=%#x error2=%#x\n", 4118 sli_reg_read(sli4, SLI4_REG_SLIPORT_STATUS), 4119 sli_reg_read(sli4, SLI4_REG_SLIPORT_ERROR1), 4120 sli_reg_read(sli4, SLI4_REG_SLIPORT_ERROR2)); 4121 } 4122 4123 /* 4124 * set the ASIC type and revision 4125 */ 4126 pci_class_rev = ocs_config_read32(os, SLI4_PCI_CLASS_REVISION); 4127 rev_id = sli_pci_rev_id(pci_class_rev); 4128 family = sli4->sli_family; 4129 if (family == SLI4_FAMILY_CHECK_ASIC_TYPE) { 4130 uint32_t asic_id = ocs_config_read32(os, SLI4_ASIC_ID_REG); 4131 family = sli_asic_gen(asic_id); 4132 } 4133 4134 for (i = 0, asic = sli4_asic_table; i < ARRAY_SIZE(sli4_asic_table); i++, asic++) { 4135 if ((rev_id == asic->rev_id) && (family == asic->family)) { 4136 sli4->asic_type = asic->type; 4137 sli4->asic_rev = asic->rev; 4138 break; 4139 } 4140 } 4141 /* Fail if no matching asic type/rev was found */ 4142 if( (sli4->asic_type == 0) || (sli4->asic_rev == 0)) { 4143 ocs_log_err(os, "no matching asic family/rev found: %02x/%02x\n", family, rev_id); 4144 return -1; 4145 } 4146 4147 /* 4148 * The bootstrap mailbox is equivalent to a MQ with a single 256 byte 4149 * entry, a CQ with a single 16 byte entry, and no event queue. 4150 * Alignment must be 16 bytes as the low order address bits in the 4151 * address register are also control / status. 4152 */ 4153 if (ocs_dma_alloc(sli4->os, &sli4->bmbx, SLI4_BMBX_SIZE + 4154 sizeof(sli4_mcqe_t), 16)) { 4155 ocs_log_err(os, "bootstrap mailbox allocation failed\n"); 4156 return -1; 4157 } 4158 4159 if (sli4->bmbx.phys & SLI4_BMBX_MASK_LO) { 4160 ocs_log_err(os, "bad alignment for bootstrap mailbox\n"); 4161 return -1; 4162 } 4163 4164 ocs_log_debug(os, "bmbx v=%p p=0x%x %08x s=%zd\n", sli4->bmbx.virt, 4165 ocs_addr32_hi(sli4->bmbx.phys), 4166 ocs_addr32_lo(sli4->bmbx.phys), 4167 sli4->bmbx.size); 4168 4169 /* TODO 4096 is arbitrary. What should this value actually be? */ 4170 if (ocs_dma_alloc(sli4->os, &sli4->vpd.data, 4096/*TODO*/, 4096)) { 4171 /* Note that failure isn't fatal in this specific case */ 4172 sli4->vpd.data.size = 0; 4173 ocs_log_test(os, "VPD buffer allocation failed\n"); 4174 } 4175 4176 if (sli_fw_init(sli4)) { 4177 ocs_log_err(sli4->os, "FW initialization failed\n"); 4178 return -1; 4179 } 4180 4181 /* 4182 * Set one of fcpi(initiator), fcpt(target), fcpc(combined) to true 4183 * in addition to any other desired features 4184 */ 4185 sli4->config.features.flag.iaab = TRUE; 4186 sli4->config.features.flag.npiv = TRUE; 4187 sli4->config.features.flag.dif = TRUE; 4188 sli4->config.features.flag.vf = TRUE; 4189 sli4->config.features.flag.fcpc = TRUE; 4190 sli4->config.features.flag.iaar = TRUE; 4191 sli4->config.features.flag.hlm = TRUE; 4192 sli4->config.features.flag.perfh = TRUE; 4193 sli4->config.features.flag.rxseq = TRUE; 4194 sli4->config.features.flag.rxri = TRUE; 4195 sli4->config.features.flag.mrqp = TRUE; 4196 4197 /* use performance hints if available */ 4198 if (sli4->config.perf_hint) { 4199 sli4->config.features.flag.perfh = TRUE; 4200 } 4201 4202 if (sli_request_features(sli4, &sli4->config.features, TRUE)) { 4203 return -1; 4204 } 4205 4206 if (sli_get_config(sli4)) { 4207 return -1; 4208 } 4209 4210 return 0; 4211 } 4212 4213 bool 4214 sli_persist_topology_enabled(sli4_t *sli4) 4215 { 4216 return (sli4->config.ptv); 4217 } 4218 4219 int32_t 4220 sli_init(sli4_t *sli4) 4221 { 4222 4223 if (sli4->config.has_extents) { 4224 /* TODO COMMON_ALLOC_RESOURCE_EXTENTS */; 4225 ocs_log_test(sli4->os, "XXX need to implement extent allocation\n"); 4226 return -1; 4227 } 4228 4229 sli4->config.features.flag.hlm = sli4->config.high_login_mode; 4230 sli4->config.features.flag.rxseq = FALSE; 4231 sli4->config.features.flag.rxri = FALSE; 4232 4233 if (sli_request_features(sli4, &sli4->config.features, FALSE)) { 4234 return -1; 4235 } 4236 4237 return 0; 4238 } 4239 4240 int32_t 4241 sli_reset(sli4_t *sli4) 4242 { 4243 uint32_t i; 4244 4245 if (sli_fw_init(sli4)) { 4246 ocs_log_crit(sli4->os, "FW initialization failed\n"); 4247 return -1; 4248 } 4249 4250 if (sli4->config.extent[0].base) { 4251 ocs_free(sli4->os, sli4->config.extent[0].base, SLI_RSRC_MAX * sizeof(uint32_t)); 4252 sli4->config.extent[0].base = NULL; 4253 } 4254 4255 for (i = 0; i < SLI_RSRC_MAX; i++) { 4256 if (sli4->config.extent[i].use_map) { 4257 ocs_bitmap_free(sli4->config.extent[i].use_map); 4258 sli4->config.extent[i].use_map = NULL; 4259 } 4260 sli4->config.extent[i].base = NULL; 4261 } 4262 4263 if (sli_get_config(sli4)) { 4264 return -1; 4265 } 4266 4267 return 0; 4268 } 4269 4270 /** 4271 * @ingroup sli 4272 * @brief Issue a Firmware Reset. 4273 * 4274 * @par Description 4275 * Issues a Firmware Reset to the chip. This reset affects the entire chip, 4276 * so all PCI function on the same PCI bus and device are affected. 4277 * @n @n This type of reset can be used to activate newly downloaded firmware. 4278 * @n @n The driver should be considered to be in an unknown state after this 4279 * reset and should be reloaded. 4280 * 4281 * @param sli4 SLI context. 4282 * 4283 * @return Returns 0 on success, or -1 otherwise. 4284 */ 4285 4286 int32_t 4287 sli_fw_reset(sli4_t *sli4) 4288 { 4289 uint32_t val; 4290 uint32_t ready; 4291 4292 /* 4293 * Firmware must be ready before issuing the reset. 4294 */ 4295 ready = sli_wait_for_fw_ready(sli4, SLI4_FW_READY_TIMEOUT_MSEC); 4296 if (!ready) { 4297 ocs_log_crit(sli4->os, "FW status is NOT ready\n"); 4298 return -1; 4299 } 4300 switch(sli4->if_type) { 4301 case SLI4_IF_TYPE_BE3_SKH_PF: 4302 /* BE3 / Skyhawk use PCICFG_SOFT_RESET_CSR */ 4303 val = ocs_config_read32(sli4->os, SLI4_PCI_SOFT_RESET_CSR); 4304 val |= SLI4_PCI_SOFT_RESET_MASK; 4305 ocs_config_write32(sli4->os, SLI4_PCI_SOFT_RESET_CSR, val); 4306 break; 4307 case SLI4_IF_TYPE_LANCER_FC_ETH: 4308 /* Lancer uses PHYDEV_CONTROL */ 4309 4310 val = SLI4_PHYDEV_CONTROL_FRST; 4311 sli_reg_write(sli4, SLI4_REG_PHYSDEV_CONTROL, val); 4312 break; 4313 default: 4314 ocs_log_test(sli4->os, "Unexpected iftype %d\n", sli4->if_type); 4315 return -1; 4316 break; 4317 } 4318 4319 /* wait for the FW to become ready after the reset */ 4320 ready = sli_wait_for_fw_ready(sli4, SLI4_FW_READY_TIMEOUT_MSEC); 4321 if (!ready) { 4322 ocs_log_crit(sli4->os, "Failed to become ready after firmware reset\n"); 4323 return -1; 4324 } 4325 return 0; 4326 } 4327 4328 /** 4329 * @ingroup sli 4330 * @brief Tear down a SLI context. 4331 * 4332 * @param sli4 SLI context. 4333 * 4334 * @return Returns 0 on success, or non-zero otherwise. 4335 */ 4336 int32_t 4337 sli_teardown(sli4_t *sli4) 4338 { 4339 uint32_t i; 4340 4341 if (sli4->config.extent[0].base) { 4342 ocs_free(sli4->os, sli4->config.extent[0].base, SLI_RSRC_MAX * sizeof(uint32_t)); 4343 sli4->config.extent[0].base = NULL; 4344 } 4345 4346 for (i = 0; i < SLI_RSRC_MAX; i++) { 4347 if (sli4->config.has_extents) { 4348 /* TODO COMMON_DEALLOC_RESOURCE_EXTENTS */; 4349 } 4350 4351 sli4->config.extent[i].base = NULL; 4352 4353 ocs_bitmap_free(sli4->config.extent[i].use_map); 4354 sli4->config.extent[i].use_map = NULL; 4355 } 4356 4357 if (sli_fw_term(sli4)) { 4358 ocs_log_err(sli4->os, "FW deinitialization failed\n"); 4359 } 4360 4361 ocs_dma_free(sli4->os, &sli4->vpd.data); 4362 ocs_dma_free(sli4->os, &sli4->bmbx); 4363 4364 return 0; 4365 } 4366 4367 /** 4368 * @ingroup sli 4369 * @brief Register a callback for the given event. 4370 * 4371 * @param sli4 SLI context. 4372 * @param which Event of interest. 4373 * @param func Function to call when the event occurs. 4374 * @param arg Argument passed to the callback function. 4375 * 4376 * @return Returns 0 on success, or non-zero otherwise. 4377 */ 4378 int32_t 4379 sli_callback(sli4_t *sli4, sli4_callback_e which, void *func, void *arg) 4380 { 4381 4382 if (!sli4 || !func || (which >= SLI4_CB_MAX)) { 4383 ocs_log_err(NULL, "bad parameter sli4=%p which=%#x func=%p\n", 4384 sli4, which, func); 4385 return -1; 4386 } 4387 4388 switch (which) { 4389 case SLI4_CB_LINK: 4390 sli4->link = func; 4391 sli4->link_arg = arg; 4392 break; 4393 case SLI4_CB_FIP: 4394 sli4->fip = func; 4395 sli4->fip_arg = arg; 4396 break; 4397 default: 4398 ocs_log_test(sli4->os, "unknown callback %#x\n", which); 4399 return -1; 4400 } 4401 4402 return 0; 4403 } 4404 4405 /** 4406 * @ingroup sli 4407 * @brief Initialize a queue object. 4408 * 4409 * @par Description 4410 * This initializes the sli4_queue_t object members, including the underlying 4411 * DMA memory. 4412 * 4413 * @param sli4 SLI context. 4414 * @param q Pointer to queue object. 4415 * @param qtype Type of queue to create. 4416 * @param size Size of each entry. 4417 * @param n_entries Number of entries to allocate. 4418 * @param align Starting memory address alignment. 4419 * 4420 * @note Checks if using the existing DMA memory (if any) is possible. If not, 4421 * it frees the existing memory and re-allocates. 4422 * 4423 * @return Returns 0 on success, or non-zero otherwise. 4424 */ 4425 int32_t 4426 __sli_queue_init(sli4_t *sli4, sli4_queue_t *q, uint32_t qtype, 4427 size_t size, uint32_t n_entries, uint32_t align) 4428 { 4429 4430 if ((q->dma.virt == NULL) || (size != q->size) || (n_entries != q->length)) { 4431 if (q->dma.size) { 4432 ocs_dma_free(sli4->os, &q->dma); 4433 } 4434 4435 ocs_memset(q, 0, sizeof(sli4_queue_t)); 4436 4437 if (ocs_dma_alloc(sli4->os, &q->dma, size * n_entries, align)) { 4438 ocs_log_err(sli4->os, "%s allocation failed\n", SLI_QNAME[qtype]); 4439 return -1; 4440 } 4441 4442 ocs_memset(q->dma.virt, 0, size * n_entries); 4443 4444 ocs_lock_init(sli4->os, &q->lock, "%s lock[%d:%p]", 4445 SLI_QNAME[qtype], ocs_instance(sli4->os), &q->lock); 4446 4447 q->type = qtype; 4448 q->size = size; 4449 q->length = n_entries; 4450 4451 /* Limit to hwf the queue size per interrupt */ 4452 q->proc_limit = n_entries / 2; 4453 4454 if ( (q->type == SLI_QTYPE_EQ) || (q->type == SLI_QTYPE_CQ) ) { 4455 /* For prism, phase will be flipped after a sweep through eq and cq */ 4456 q->phase = 1; 4457 } 4458 4459 switch(q->type) { 4460 case SLI_QTYPE_EQ: 4461 q->posted_limit = q->length / 2; 4462 break; 4463 default: 4464 if ((sli4->if_type == SLI4_IF_TYPE_BE3_SKH_PF) || 4465 (sli4->if_type == SLI4_IF_TYPE_BE3_SKH_VF)) { 4466 /* For Skyhawk, ring the doorbell more often */ 4467 q->posted_limit = 8; 4468 } else { 4469 q->posted_limit = 64; 4470 } 4471 break; 4472 } 4473 } 4474 4475 return 0; 4476 } 4477 4478 /** 4479 * @ingroup sli 4480 * @brief Issue the command to create a queue. 4481 * 4482 * @param sli4 SLI context. 4483 * @param q Pointer to queue object. 4484 * 4485 * @return Returns 0 on success, or non-zero otherwise. 4486 */ 4487 int32_t 4488 __sli_create_queue(sli4_t *sli4, sli4_queue_t *q) 4489 { 4490 sli4_res_common_create_queue_t *res_q = NULL; 4491 4492 if (sli_bmbx_command(sli4)){ 4493 ocs_log_crit(sli4->os, "bootstrap mailbox write fail %s\n", 4494 SLI_QNAME[q->type]); 4495 ocs_dma_free(sli4->os, &q->dma); 4496 return -1; 4497 } 4498 if (sli_res_sli_config(sli4->bmbx.virt)) { 4499 ocs_log_err(sli4->os, "bad status create %s\n", SLI_QNAME[q->type]); 4500 ocs_dma_free(sli4->os, &q->dma); 4501 return -1; 4502 } 4503 res_q = (void *)((uint8_t *)sli4->bmbx.virt + 4504 offsetof(sli4_cmd_sli_config_t, payload)); 4505 4506 if (res_q->hdr.status) { 4507 ocs_log_err(sli4->os, "bad create %s status=%#x addl=%#x\n", 4508 SLI_QNAME[q->type], 4509 res_q->hdr.status, res_q->hdr.additional_status); 4510 ocs_dma_free(sli4->os, &q->dma); 4511 return -1; 4512 } else { 4513 q->id = res_q->q_id; 4514 q->doorbell_offset = res_q->db_offset; 4515 q->doorbell_rset = res_q->db_rs; 4516 4517 switch (q->type) { 4518 case SLI_QTYPE_EQ: 4519 /* No doorbell information in response for EQs */ 4520 q->doorbell_offset = regmap[SLI4_REG_EQ_DOORBELL][sli4->if_type].off; 4521 q->doorbell_rset = regmap[SLI4_REG_EQ_DOORBELL][sli4->if_type].rset; 4522 break; 4523 case SLI_QTYPE_CQ: 4524 /* No doorbell information in response for CQs */ 4525 q->doorbell_offset = regmap[SLI4_REG_CQ_DOORBELL][sli4->if_type].off; 4526 q->doorbell_rset = regmap[SLI4_REG_CQ_DOORBELL][sli4->if_type].rset; 4527 break; 4528 case SLI_QTYPE_MQ: 4529 /* No doorbell information in response for MQs */ 4530 q->doorbell_offset = regmap[SLI4_REG_MQ_DOORBELL][sli4->if_type].off; 4531 q->doorbell_rset = regmap[SLI4_REG_MQ_DOORBELL][sli4->if_type].rset; 4532 break; 4533 case SLI_QTYPE_RQ: 4534 /* set the doorbell for non-skyhawks */ 4535 if (!sli4->config.dual_ulp_capable) { 4536 q->doorbell_offset = regmap[SLI4_REG_FCOE_RQ_DOORBELL][sli4->if_type].off; 4537 q->doorbell_rset = regmap[SLI4_REG_FCOE_RQ_DOORBELL][sli4->if_type].rset; 4538 } 4539 break; 4540 case SLI_QTYPE_WQ: 4541 /* set the doorbell for non-skyhawks */ 4542 if (!sli4->config.dual_ulp_capable) { 4543 q->doorbell_offset = regmap[SLI4_REG_IO_WQ_DOORBELL][sli4->if_type].off; 4544 q->doorbell_rset = regmap[SLI4_REG_IO_WQ_DOORBELL][sli4->if_type].rset; 4545 } 4546 break; 4547 default: 4548 break; 4549 } 4550 } 4551 4552 return 0; 4553 } 4554 4555 /** 4556 * @ingroup sli 4557 * @brief Get queue entry size. 4558 * 4559 * Get queue entry size given queue type. 4560 * 4561 * @param sli4 SLI context 4562 * @param qtype Type for which the entry size is returned. 4563 * 4564 * @return Returns > 0 on success (queue entry size), or a negative value on failure. 4565 */ 4566 int32_t 4567 sli_get_queue_entry_size(sli4_t *sli4, uint32_t qtype) 4568 { 4569 uint32_t size = 0; 4570 4571 if (!sli4) { 4572 ocs_log_err(NULL, "bad parameter sli4=%p\n", sli4); 4573 return -1; 4574 } 4575 4576 switch (qtype) { 4577 case SLI_QTYPE_EQ: 4578 size = sizeof(uint32_t); 4579 break; 4580 case SLI_QTYPE_CQ: 4581 size = 16; 4582 break; 4583 case SLI_QTYPE_MQ: 4584 size = 256; 4585 break; 4586 case SLI_QTYPE_WQ: 4587 if (SLI4_PORT_TYPE_FC == sli4->port_type) { 4588 size = sli4->config.wqe_size; 4589 } else { 4590 /* TODO */ 4591 ocs_log_test(sli4->os, "unsupported queue entry size\n"); 4592 return -1; 4593 } 4594 break; 4595 case SLI_QTYPE_RQ: 4596 size = SLI4_FCOE_RQE_SIZE; 4597 break; 4598 default: 4599 ocs_log_test(sli4->os, "unknown queue type %d\n", qtype); 4600 return -1; 4601 } 4602 return size; 4603 } 4604 4605 /** 4606 * @ingroup sli 4607 * @brief Modify the delay timer for all the EQs 4608 * 4609 * @param sli4 SLI context. 4610 * @param eq Array of EQs. 4611 * @param num_eq Count of EQs. 4612 * @param shift Phase shift for staggering interrupts. 4613 * @param delay_mult Delay multiplier for limiting interrupt frequency. 4614 * 4615 * @return Returns 0 on success, or -1 otherwise. 4616 */ 4617 int32_t 4618 sli_eq_modify_delay(sli4_t *sli4, sli4_queue_t *eq, uint32_t num_eq, uint32_t shift, uint32_t delay_mult) 4619 { 4620 4621 sli_cmd_common_modify_eq_delay(sli4, sli4->bmbx.virt, SLI4_BMBX_SIZE, eq, num_eq, shift, delay_mult); 4622 4623 if (sli_bmbx_command(sli4)) { 4624 ocs_log_crit(sli4->os, "bootstrap mailbox write fail (MODIFY EQ DELAY)\n"); 4625 return -1; 4626 } 4627 if (sli_res_sli_config(sli4->bmbx.virt)) { 4628 ocs_log_err(sli4->os, "bad status MODIFY EQ DELAY\n"); 4629 return -1; 4630 } 4631 4632 return 0; 4633 } 4634 4635 /** 4636 * @ingroup sli 4637 * @brief Allocate a queue. 4638 * 4639 * @par Description 4640 * Allocates DMA memory and configures the requested queue type. 4641 * 4642 * @param sli4 SLI context. 4643 * @param qtype Type of queue to create. 4644 * @param q Pointer to the queue object. 4645 * @param n_entries Number of entries to allocate. 4646 * @param assoc Associated queue (that is, the EQ for a CQ, the CQ for a MQ, and so on). 4647 * @param ulp The ULP to bind, which is only used for WQ and RQs 4648 * 4649 * @return Returns 0 on success, or -1 otherwise. 4650 */ 4651 int32_t 4652 sli_queue_alloc(sli4_t *sli4, uint32_t qtype, sli4_queue_t *q, uint32_t n_entries, 4653 sli4_queue_t *assoc, uint16_t ulp) 4654 { 4655 int32_t size; 4656 uint32_t align = 0; 4657 sli4_create_q_fn_t create = NULL; 4658 4659 if (!sli4 || !q) { 4660 ocs_log_err(NULL, "bad parameter sli4=%p q=%p\n", sli4, q); 4661 return -1; 4662 } 4663 4664 /* get queue size */ 4665 size = sli_get_queue_entry_size(sli4, qtype); 4666 if (size < 0) 4667 return -1; 4668 align = SLI_PAGE_SIZE; 4669 4670 switch (qtype) { 4671 case SLI_QTYPE_EQ: 4672 create = sli_cmd_common_create_eq; 4673 break; 4674 case SLI_QTYPE_CQ: 4675 create = sli_cmd_common_create_cq; 4676 break; 4677 case SLI_QTYPE_MQ: 4678 /* Validate the number of entries */ 4679 switch (n_entries) { 4680 case 16: 4681 case 32: 4682 case 64: 4683 case 128: 4684 break; 4685 default: 4686 ocs_log_test(sli4->os, "illegal n_entries value %d for MQ\n", n_entries); 4687 return -1; 4688 } 4689 assoc->u.flag.is_mq = TRUE; 4690 create = sli_cmd_common_create_mq_ext; 4691 break; 4692 case SLI_QTYPE_WQ: 4693 if (SLI4_PORT_TYPE_FC == sli4->port_type) { 4694 if (sli4->if_type == SLI4_IF_TYPE_BE3_SKH_PF) { 4695 create = sli_cmd_fcoe_wq_create; 4696 } else { 4697 create = sli_cmd_fcoe_wq_create_v1; 4698 } 4699 } else { 4700 /* TODO */ 4701 ocs_log_test(sli4->os, "unsupported WQ create\n"); 4702 return -1; 4703 } 4704 break; 4705 default: 4706 ocs_log_test(sli4->os, "unknown queue type %d\n", qtype); 4707 return -1; 4708 } 4709 4710 if (__sli_queue_init(sli4, q, qtype, size, n_entries, align)) { 4711 ocs_log_err(sli4->os, "%s allocation failed\n", SLI_QNAME[qtype]); 4712 return -1; 4713 } 4714 4715 if (create(sli4, sli4->bmbx.virt, SLI4_BMBX_SIZE, &q->dma, assoc ? assoc->id : 0, ulp)) { 4716 if (__sli_create_queue(sli4, q)) { 4717 ocs_log_err(sli4->os, "create %s failed\n", SLI_QNAME[qtype]); 4718 return -1; 4719 } 4720 q->ulp = ulp; 4721 } else { 4722 ocs_log_err(sli4->os, "cannot create %s\n", SLI_QNAME[qtype]); 4723 return -1; 4724 } 4725 4726 return 0; 4727 } 4728 4729 /** 4730 * @ingroup sli 4731 * @brief Allocate a c queue set. 4732 * 4733 * @param sli4 SLI context. 4734 * @param num_cqs to create 4735 * @param qs Pointers to the queue objects. 4736 * @param n_entries Number of entries to allocate per CQ. 4737 * @param eqs Associated event queues 4738 * 4739 * @return Returns 0 on success, or -1 otherwise. 4740 */ 4741 int32_t 4742 sli_cq_alloc_set(sli4_t *sli4, sli4_queue_t *qs[], uint32_t num_cqs, 4743 uint32_t n_entries, sli4_queue_t *eqs[]) 4744 { 4745 uint32_t i, offset = 0, page_bytes = 0, payload_size, cmd_size = 0; 4746 uint32_t p = 0, page_size = 0, n_cqe = 0, num_pages_cq; 4747 uintptr_t addr; 4748 ocs_dma_t dma; 4749 sli4_req_common_create_cq_set_v0_t *req = NULL; 4750 sli4_res_common_create_queue_set_t *res = NULL; 4751 4752 if (!sli4) { 4753 ocs_log_err(NULL, "bad parameter sli4=%p\n", sli4); 4754 return -1; 4755 } 4756 4757 memset(&dma, 0, sizeof(dma)); 4758 4759 /* Align the queue DMA memory */ 4760 for (i = 0; i < num_cqs; i++) { 4761 if (__sli_queue_init(sli4, qs[i], SLI_QTYPE_CQ, SLI4_CQE_BYTES, 4762 n_entries, SLI_PAGE_SIZE)) { 4763 ocs_log_err(sli4->os, "Queue init failed.\n"); 4764 goto error; 4765 } 4766 } 4767 4768 n_cqe = qs[0]->dma.size / SLI4_CQE_BYTES; 4769 switch (n_cqe) { 4770 case 256: 4771 case 512: 4772 case 1024: 4773 case 2048: 4774 page_size = 1; 4775 break; 4776 case 4096: 4777 page_size = 2; 4778 break; 4779 default: 4780 return -1; 4781 } 4782 4783 page_bytes = page_size * SLI_PAGE_SIZE; 4784 num_pages_cq = sli_page_count(qs[0]->dma.size, page_bytes); 4785 cmd_size = sizeof(sli4_req_common_create_cq_set_v0_t) + (8 * num_pages_cq * num_cqs); 4786 payload_size = max((size_t)cmd_size, sizeof(sli4_res_common_create_queue_set_t)); 4787 4788 if (ocs_dma_alloc(sli4->os, &dma, payload_size, SLI_PAGE_SIZE)) { 4789 ocs_log_err(sli4->os, "DMA allocation failed\n"); 4790 goto error; 4791 } 4792 ocs_memset(dma.virt, 0, payload_size); 4793 4794 if (sli_cmd_sli_config(sli4, sli4->bmbx.virt, SLI4_BMBX_SIZE, 4795 payload_size, &dma) == -1) { 4796 goto error; 4797 } 4798 4799 /* Fill the request structure */ 4800 4801 req = (sli4_req_common_create_cq_set_v0_t *)((uint8_t *)dma.virt); 4802 req->hdr.opcode = SLI4_OPC_COMMON_CREATE_CQ_SET; 4803 req->hdr.subsystem = SLI4_SUBSYSTEM_FCFCOE; 4804 req->hdr.version = 0; 4805 req->hdr.request_length = cmd_size - sizeof(sli4_req_hdr_t); 4806 req->page_size = page_size; 4807 4808 req->num_pages = num_pages_cq; 4809 switch (req->num_pages) { 4810 case 1: 4811 req->cqecnt = SLI4_CQ_CNT_256; 4812 break; 4813 case 2: 4814 req->cqecnt = SLI4_CQ_CNT_512; 4815 break; 4816 case 4: 4817 req->cqecnt = SLI4_CQ_CNT_1024; 4818 break; 4819 case 8: 4820 req->cqecnt = SLI4_CQ_CNT_LARGE; 4821 req->cqe_count = n_cqe; 4822 break; 4823 default: 4824 ocs_log_test(sli4->os, "num_pages %d not valid\n", req->num_pages); 4825 goto error; 4826 } 4827 4828 req->evt = TRUE; 4829 req->valid = TRUE; 4830 req->arm = FALSE; 4831 req->num_cq_req = num_cqs; 4832 4833 if (sli4->if_type == SLI4_IF_TYPE_LANCER_G7) 4834 req->autovalid = TRUE; 4835 4836 /* Fill page addresses of all the CQs. */ 4837 for (i = 0; i < num_cqs; i++) { 4838 req->eq_id[i] = eqs[i]->id; 4839 for (p = 0, addr = qs[i]->dma.phys; p < req->num_pages; p++, addr += page_bytes) { 4840 req->page_physical_address[offset].low = ocs_addr32_lo(addr); 4841 req->page_physical_address[offset].high = ocs_addr32_hi(addr); 4842 offset++; 4843 } 4844 } 4845 4846 if (sli_bmbx_command(sli4)) { 4847 ocs_log_crit(sli4->os, "bootstrap mailbox write fail CQSet\n"); 4848 goto error; 4849 } 4850 4851 res = (void *)((uint8_t *)dma.virt); 4852 if (res->hdr.status) { 4853 ocs_log_err(sli4->os, "bad create CQSet status=%#x addl=%#x\n", 4854 res->hdr.status, res->hdr.additional_status); 4855 goto error; 4856 } else { 4857 /* Check if we got all requested CQs. */ 4858 if (res->num_q_allocated != num_cqs) { 4859 ocs_log_crit(sli4->os, "Requested count CQs doesnt match.\n"); 4860 goto error; 4861 } 4862 4863 /* Fill the resp cq ids. */ 4864 for (i = 0; i < num_cqs; i++) { 4865 qs[i]->id = res->q_id + i; 4866 qs[i]->doorbell_offset = regmap[SLI4_REG_CQ_DOORBELL][sli4->if_type].off; 4867 qs[i]->doorbell_rset = regmap[SLI4_REG_CQ_DOORBELL][sli4->if_type].rset; 4868 } 4869 } 4870 4871 ocs_dma_free(sli4->os, &dma); 4872 4873 return 0; 4874 4875 error: 4876 for (i = 0; i < num_cqs; i++) { 4877 if (qs[i]->dma.size) { 4878 ocs_dma_free(sli4->os, &qs[i]->dma); 4879 } 4880 } 4881 4882 if (dma.size) { 4883 ocs_dma_free(sli4->os, &dma); 4884 } 4885 4886 return -1; 4887 } 4888 4889 /** 4890 * @ingroup sli 4891 * @brief Free a queue. 4892 * 4893 * @par Description 4894 * Frees DMA memory and de-registers the requested queue. 4895 * 4896 * @param sli4 SLI context. 4897 * @param q Pointer to the queue object. 4898 * @param destroy_queues Non-zero if the mailbox commands should be sent to destroy the queues. 4899 * @param free_memory Non-zero if the DMA memory associated with the queue should be freed. 4900 * 4901 * @return Returns 0 on success, or -1 otherwise. 4902 */ 4903 int32_t 4904 sli_queue_free(sli4_t *sli4, sli4_queue_t *q, uint32_t destroy_queues, uint32_t free_memory) 4905 { 4906 sli4_destroy_q_fn_t destroy = NULL; 4907 int32_t rc = -1; 4908 4909 if (!sli4 || !q) { 4910 ocs_log_err(NULL, "bad parameter sli4=%p q=%p\n", sli4, q); 4911 return -1; 4912 } 4913 4914 if (destroy_queues) { 4915 switch (q->type) { 4916 case SLI_QTYPE_EQ: 4917 destroy = sli_cmd_common_destroy_eq; 4918 break; 4919 case SLI_QTYPE_CQ: 4920 destroy = sli_cmd_common_destroy_cq; 4921 break; 4922 case SLI_QTYPE_MQ: 4923 destroy = sli_cmd_common_destroy_mq; 4924 break; 4925 case SLI_QTYPE_WQ: 4926 if (SLI4_PORT_TYPE_FC == sli4->port_type) { 4927 destroy = sli_cmd_fcoe_wq_destroy; 4928 } else { 4929 /* TODO */ 4930 ocs_log_test(sli4->os, "unsupported WQ destroy\n"); 4931 return -1; 4932 } 4933 break; 4934 case SLI_QTYPE_RQ: 4935 if (SLI4_PORT_TYPE_FC == sli4->port_type) { 4936 destroy = sli_cmd_fcoe_rq_destroy; 4937 } else { 4938 /* TODO */ 4939 ocs_log_test(sli4->os, "unsupported RQ destroy\n"); 4940 return -1; 4941 } 4942 break; 4943 default: 4944 ocs_log_test(sli4->os, "bad queue type %d\n", 4945 q->type); 4946 return -1; 4947 } 4948 4949 /* 4950 * Destroying queues makes BE3 sad (version 0 interface type). Rely 4951 * on COMMON_FUNCTION_RESET to free host allocated queue resources 4952 * inside the SLI Port. 4953 */ 4954 if (SLI4_IF_TYPE_BE3_SKH_PF == sli4->if_type) { 4955 destroy = NULL; 4956 } 4957 4958 /* Destroy the queue if the operation is defined */ 4959 if (destroy && destroy(sli4, sli4->bmbx.virt, SLI4_BMBX_SIZE, q->id)) { 4960 sli4_res_hdr_t *res = NULL; 4961 4962 if (sli_bmbx_command(sli4)){ 4963 ocs_log_crit(sli4->os, "bootstrap mailbox write fail destroy %s\n", 4964 SLI_QNAME[q->type]); 4965 } else if (sli_res_sli_config(sli4->bmbx.virt)) { 4966 ocs_log_err(sli4->os, "bad status destroy %s\n", SLI_QNAME[q->type]); 4967 } else { 4968 res = (void *)((uint8_t *)sli4->bmbx.virt + 4969 offsetof(sli4_cmd_sli_config_t, payload)); 4970 4971 if (res->status) { 4972 ocs_log_err(sli4->os, "bad destroy %s status=%#x addl=%#x\n", 4973 SLI_QNAME[q->type], 4974 res->status, res->additional_status); 4975 } else { 4976 rc = 0; 4977 } 4978 } 4979 } 4980 } 4981 4982 if (free_memory) { 4983 ocs_lock_free(&q->lock); 4984 4985 if (ocs_dma_free(sli4->os, &q->dma)) { 4986 ocs_log_err(sli4->os, "%s queue ID %d free failed\n", 4987 SLI_QNAME[q->type], q->id); 4988 rc = -1; 4989 } 4990 } 4991 4992 return rc; 4993 } 4994 4995 int32_t 4996 sli_queue_reset(sli4_t *sli4, sli4_queue_t *q) 4997 { 4998 4999 ocs_lock(&q->lock); 5000 5001 q->index = 0; 5002 q->n_posted = 0; 5003 5004 if (SLI_QTYPE_MQ == q->type) { 5005 q->u.r_idx = 0; 5006 } 5007 5008 if (q->dma.virt != NULL) { 5009 ocs_memset(q->dma.virt, 0, (q->size * (uint64_t)q->length)); 5010 } 5011 5012 ocs_unlock(&q->lock); 5013 5014 return 0; 5015 } 5016 5017 /** 5018 * @ingroup sli 5019 * @brief Check if the given queue is empty. 5020 * 5021 * @par Description 5022 * If the valid bit of the current entry is unset, the queue is empty. 5023 * 5024 * @param sli4 SLI context. 5025 * @param q Pointer to the queue object. 5026 * 5027 * @return Returns TRUE if empty, or FALSE otherwise. 5028 */ 5029 int32_t 5030 sli_queue_is_empty(sli4_t *sli4, sli4_queue_t *q) 5031 { 5032 int32_t rc = TRUE; 5033 uint8_t *qe = q->dma.virt; 5034 5035 ocs_lock(&q->lock); 5036 5037 ocs_dma_sync(&q->dma, OCS_DMASYNC_POSTREAD); 5038 5039 qe += q->index * q->size; 5040 5041 rc = !sli_queue_entry_is_valid(q, qe, FALSE); 5042 5043 ocs_unlock(&q->lock); 5044 5045 return rc; 5046 } 5047 5048 /** 5049 * @ingroup sli 5050 * @brief Arm an EQ. 5051 * 5052 * @param sli4 SLI context. 5053 * @param q Pointer to queue object. 5054 * @param arm If TRUE, arm the EQ. 5055 * 5056 * @return Returns 0 on success, or non-zero otherwise. 5057 */ 5058 int32_t 5059 sli_queue_eq_arm(sli4_t *sli4, sli4_queue_t *q, uint8_t arm) 5060 { 5061 uint32_t val = 0; 5062 5063 ocs_lock(&q->lock); 5064 if (sli4->if_type == SLI4_IF_TYPE_LANCER_G7) 5065 val = sli_iftype6_eq_doorbell(q->n_posted, q->id, arm); 5066 else 5067 val = sli_eq_doorbell(q->n_posted, q->id, arm); 5068 5069 ocs_reg_write32(sli4->os, q->doorbell_rset, q->doorbell_offset, val); 5070 q->n_posted = 0; 5071 ocs_unlock(&q->lock); 5072 5073 return 0; 5074 } 5075 5076 /** 5077 * @ingroup sli 5078 * @brief Arm a queue. 5079 * 5080 * @param sli4 SLI context. 5081 * @param q Pointer to queue object. 5082 * @param arm If TRUE, arm the queue. 5083 * 5084 * @return Returns 0 on success, or non-zero otherwise. 5085 */ 5086 int32_t 5087 sli_queue_arm(sli4_t *sli4, sli4_queue_t *q, uint8_t arm) 5088 { 5089 uint32_t val = 0; 5090 5091 ocs_lock(&q->lock); 5092 5093 switch (q->type) { 5094 case SLI_QTYPE_EQ: 5095 if (sli4->if_type == SLI4_IF_TYPE_LANCER_G7) 5096 val = sli_iftype6_eq_doorbell(q->n_posted, q->id, arm); 5097 else 5098 val = sli_eq_doorbell(q->n_posted, q->id, arm); 5099 ocs_reg_write32(sli4->os, q->doorbell_rset, q->doorbell_offset, val); 5100 q->n_posted = 0; 5101 break; 5102 case SLI_QTYPE_CQ: 5103 if (sli4->if_type == SLI4_IF_TYPE_LANCER_G7) 5104 val = sli_iftype6_cq_doorbell(q->n_posted, q->id, arm); 5105 else 5106 val = sli_cq_doorbell(q->n_posted, q->id, arm); 5107 ocs_reg_write32(sli4->os, q->doorbell_rset, q->doorbell_offset, val); 5108 q->n_posted = 0; 5109 break; 5110 default: 5111 ocs_log_test(sli4->os, "should only be used for EQ/CQ, not %s\n", 5112 SLI_QNAME[q->type]); 5113 } 5114 5115 ocs_unlock(&q->lock); 5116 5117 return 0; 5118 } 5119 5120 /** 5121 * @ingroup sli 5122 * @brief Write an entry to the queue object. 5123 * 5124 * Note: Assumes the q->lock will be locked and released by the caller. 5125 * 5126 * @param sli4 SLI context. 5127 * @param q Pointer to the queue object. 5128 * @param entry Pointer to the entry contents. 5129 * 5130 * @return Returns queue index on success, or negative error value otherwise. 5131 */ 5132 int32_t 5133 _sli_queue_write(sli4_t *sli4, sli4_queue_t *q, uint8_t *entry) 5134 { 5135 int32_t rc = 0; 5136 uint8_t *qe = q->dma.virt; 5137 uint32_t qindex; 5138 5139 qindex = q->index; 5140 qe += q->index * q->size; 5141 5142 if (entry) { 5143 if ((SLI_QTYPE_WQ == q->type) && sli4->config.perf_wq_id_association) { 5144 sli_set_wq_id_association(entry, q->id); 5145 } 5146 #if defined(OCS_INCLUDE_DEBUG) 5147 switch (q->type) { 5148 case SLI_QTYPE_WQ: { 5149 ocs_dump32(OCS_DEBUG_ENABLE_WQ_DUMP, sli4->os, "wqe", entry, q->size); 5150 break; 5151 } 5152 case SLI_QTYPE_MQ: 5153 /* Note: we don't really need to dump the whole 5154 * 256 bytes, just do 64 */ 5155 ocs_dump32(OCS_DEBUG_ENABLE_MQ_DUMP, sli4->os, "mqe outbound", entry, 64); 5156 break; 5157 5158 default: 5159 break; 5160 } 5161 #endif 5162 ocs_memcpy(qe, entry, q->size); 5163 q->n_posted = 1; 5164 } 5165 5166 ocs_dma_sync(&q->dma, OCS_DMASYNC_PREWRITE); 5167 5168 rc = sli_queue_doorbell(sli4, q); 5169 5170 q->index = (q->index + q->n_posted) & (q->length - 1); 5171 q->n_posted = 0; 5172 5173 if (rc < 0) { 5174 /* failure */ 5175 return rc; 5176 } else if (rc > 0) { 5177 /* failure, but we need to return a negative value on failure */ 5178 return -rc; 5179 } else { 5180 return qindex; 5181 } 5182 } 5183 5184 /** 5185 * @ingroup sli 5186 * @brief Write an entry to the queue object. 5187 * 5188 * Note: Assumes the q->lock will be locked and released by the caller. 5189 * 5190 * @param sli4 SLI context. 5191 * @param q Pointer to the queue object. 5192 * @param entry Pointer to the entry contents. 5193 * 5194 * @return Returns queue index on success, or negative error value otherwise. 5195 */ 5196 int32_t 5197 sli_queue_write(sli4_t *sli4, sli4_queue_t *q, uint8_t *entry) 5198 { 5199 int32_t rc; 5200 5201 ocs_lock(&q->lock); 5202 rc = _sli_queue_write(sli4, q, entry); 5203 ocs_unlock(&q->lock); 5204 5205 return rc; 5206 } 5207 5208 /** 5209 * @brief Check if the current queue entry is valid. 5210 * 5211 * @param q Pointer to the queue object. 5212 * @param qe Pointer to the queue entry. 5213 * @param clear Boolean to clear valid bit. 5214 * 5215 * @return Returns TRUE if the entry is valid, or FALSE otherwise. 5216 */ 5217 static uint8_t 5218 sli_queue_entry_is_valid(sli4_queue_t *q, uint8_t *qe, uint8_t clear) 5219 { 5220 uint8_t valid = FALSE; 5221 uint8_t valid_bit_set = 0; 5222 5223 switch (q->type) { 5224 case SLI_QTYPE_EQ: 5225 valid = (((sli4_eqe_t *)qe)->vld == q->phase) ? 1 : 0; 5226 if (valid && clear) { 5227 ((sli4_eqe_t *)qe)->vld = 0; 5228 } 5229 break; 5230 case SLI_QTYPE_CQ: 5231 /* 5232 * For both MCQE and WCQE/RCQE, the valid bit 5233 * is bit 31 of dword 3 (0 based) 5234 */ 5235 valid_bit_set = (qe[15] & 0x80) != 0; 5236 if (valid_bit_set == q->phase) 5237 valid = 1; 5238 5239 if (valid & clear) { 5240 qe[15] &= ~0x80; 5241 } 5242 break; 5243 case SLI_QTYPE_MQ: 5244 valid = q->index != q->u.r_idx; 5245 break; 5246 case SLI_QTYPE_RQ: 5247 valid = TRUE; 5248 clear = FALSE; 5249 break; 5250 default: 5251 ocs_log_test(NULL, "doesn't handle type=%#x\n", q->type); 5252 } 5253 5254 if (clear) { 5255 5256 ocs_dma_sync(&q->dma, OCS_DMASYNC_PREWRITE); 5257 } 5258 5259 return valid; 5260 } 5261 5262 /** 5263 * @ingroup sli 5264 * @brief Read an entry from the queue object. 5265 * 5266 * @param sli4 SLI context. 5267 * @param q Pointer to the queue object. 5268 * @param entry Destination pointer for the queue entry contents. 5269 * 5270 * @return Returns 0 on success, or non-zero otherwise. 5271 */ 5272 int32_t 5273 sli_queue_read(sli4_t *sli4, sli4_queue_t *q, uint8_t *entry) 5274 { 5275 int32_t rc = 0; 5276 uint8_t *qe = q->dma.virt; 5277 uint32_t *qindex = NULL; 5278 5279 uint8_t clear = (SLI4_IF_TYPE_LANCER_G7 == sli_get_if_type(sli4)) ? FALSE : TRUE; 5280 if (SLI_QTYPE_MQ == q->type) { 5281 qindex = &q->u.r_idx; 5282 } else { 5283 qindex = &q->index; 5284 } 5285 5286 ocs_lock(&q->lock); 5287 5288 ocs_dma_sync(&q->dma, OCS_DMASYNC_POSTREAD); 5289 5290 qe += *qindex * q->size; 5291 5292 if (!sli_queue_entry_is_valid(q, qe, clear)) { 5293 ocs_unlock(&q->lock); 5294 return -1; 5295 } 5296 5297 if (entry) { 5298 ocs_memcpy(entry, qe, q->size); 5299 #if defined(OCS_INCLUDE_DEBUG) 5300 switch(q->type) { 5301 case SLI_QTYPE_CQ: 5302 ocs_dump32(OCS_DEBUG_ENABLE_CQ_DUMP, sli4->os, "cq", entry, q->size); 5303 break; 5304 case SLI_QTYPE_MQ: 5305 ocs_dump32(OCS_DEBUG_ENABLE_MQ_DUMP, sli4->os, "mq Compl", entry, 64); 5306 break; 5307 case SLI_QTYPE_EQ: 5308 ocs_dump32(OCS_DEBUG_ENABLE_EQ_DUMP, sli4->os, "eq Compl", entry, q->size); 5309 break; 5310 default: 5311 break; 5312 } 5313 #endif 5314 } 5315 5316 switch (q->type) { 5317 case SLI_QTYPE_EQ: 5318 case SLI_QTYPE_CQ: 5319 case SLI_QTYPE_MQ: 5320 *qindex = (*qindex + 1) & (q->length - 1); 5321 if (SLI_QTYPE_MQ != q->type) { 5322 q->n_posted++; 5323 /* 5324 * For prism, the phase value will be used to check the validity of eq/cq entries. 5325 * The value toggles after a complete sweep through the queue. 5326 */ 5327 if ((SLI4_IF_TYPE_LANCER_G7 == sli_get_if_type(sli4)) && (*qindex == 0)) { 5328 q->phase ^= (uint16_t) 0x1; 5329 } 5330 } 5331 break; 5332 default: 5333 /* reads don't update the index */ 5334 break; 5335 } 5336 5337 ocs_unlock(&q->lock); 5338 5339 return rc; 5340 } 5341 5342 int32_t 5343 sli_queue_index(sli4_t *sli4, sli4_queue_t *q) 5344 { 5345 5346 if (q) { 5347 return q->index; 5348 } else { 5349 return -1; 5350 } 5351 } 5352 5353 int32_t 5354 sli_queue_poke(sli4_t *sli4, sli4_queue_t *q, uint32_t index, uint8_t *entry) 5355 { 5356 int32_t rc; 5357 5358 ocs_lock(&q->lock); 5359 rc = _sli_queue_poke(sli4, q, index, entry); 5360 ocs_unlock(&q->lock); 5361 5362 return rc; 5363 } 5364 5365 int32_t 5366 _sli_queue_poke(sli4_t *sli4, sli4_queue_t *q, uint32_t index, uint8_t *entry) 5367 { 5368 int32_t rc = 0; 5369 uint8_t *qe = q->dma.virt; 5370 5371 if (index >= q->length) { 5372 return -1; 5373 } 5374 5375 qe += index * q->size; 5376 5377 if (entry) { 5378 ocs_memcpy(qe, entry, q->size); 5379 } 5380 5381 ocs_dma_sync(&q->dma, OCS_DMASYNC_PREWRITE); 5382 5383 return rc; 5384 } 5385 5386 /** 5387 * @ingroup sli 5388 * @brief Allocate SLI Port resources. 5389 * 5390 * @par Description 5391 * Allocate port-related resources, such as VFI, RPI, XRI, and so on. 5392 * Resources are modeled using extents, regardless of whether the underlying 5393 * device implements resource extents. If the device does not implement 5394 * extents, the SLI layer models this as a single (albeit large) extent. 5395 * 5396 * @param sli4 SLI context. 5397 * @param rtype Resource type (for example, RPI or XRI) 5398 * @param rid Allocated resource ID. 5399 * @param index Index into the bitmap. 5400 * 5401 * @return Returns 0 on success, or a non-zero value on failure. 5402 */ 5403 int32_t 5404 sli_resource_alloc(sli4_t *sli4, sli4_resource_e rtype, uint32_t *rid, uint32_t *index) 5405 { 5406 int32_t rc = 0; 5407 uint32_t size; 5408 uint32_t extent_idx; 5409 uint32_t item_idx; 5410 int status; 5411 5412 *rid = UINT32_MAX; 5413 *index = UINT32_MAX; 5414 5415 switch (rtype) { 5416 case SLI_RSRC_FCOE_VFI: 5417 case SLI_RSRC_FCOE_VPI: 5418 case SLI_RSRC_FCOE_RPI: 5419 case SLI_RSRC_FCOE_XRI: 5420 status = ocs_bitmap_find(sli4->config.extent[rtype].use_map, 5421 sli4->config.extent[rtype].map_size); 5422 if (status < 0) { 5423 ocs_log_err(sli4->os, "out of resource %d (alloc=%d)\n", 5424 rtype, sli4->config.extent[rtype].n_alloc); 5425 rc = -1; 5426 break; 5427 } else { 5428 *index = status; 5429 } 5430 5431 size = sli4->config.extent[rtype].size; 5432 5433 extent_idx = *index / size; 5434 item_idx = *index % size; 5435 5436 *rid = sli4->config.extent[rtype].base[extent_idx] + item_idx; 5437 5438 sli4->config.extent[rtype].n_alloc++; 5439 break; 5440 default: 5441 rc = -1; 5442 } 5443 5444 return rc; 5445 } 5446 5447 /** 5448 * @ingroup sli 5449 * @brief Free the SLI Port resources. 5450 * 5451 * @par Description 5452 * Free port-related resources, such as VFI, RPI, XRI, and so. See discussion of 5453 * "extent" usage in sli_resource_alloc. 5454 * 5455 * @param sli4 SLI context. 5456 * @param rtype Resource type (for example, RPI or XRI). 5457 * @param rid Allocated resource ID. 5458 * 5459 * @return Returns 0 on success, or a non-zero value on failure. 5460 */ 5461 int32_t 5462 sli_resource_free(sli4_t *sli4, sli4_resource_e rtype, uint32_t rid) 5463 { 5464 int32_t rc = -1; 5465 uint32_t x; 5466 uint32_t size, *base; 5467 5468 switch (rtype) { 5469 case SLI_RSRC_FCOE_VFI: 5470 case SLI_RSRC_FCOE_VPI: 5471 case SLI_RSRC_FCOE_RPI: 5472 case SLI_RSRC_FCOE_XRI: 5473 /* 5474 * Figure out which extent contains the resource ID. I.e. find 5475 * the extent such that 5476 * extent->base <= resource ID < extent->base + extent->size 5477 */ 5478 base = sli4->config.extent[rtype].base; 5479 size = sli4->config.extent[rtype].size; 5480 5481 /* 5482 * In the case of FW reset, this may be cleared but the force_free path will 5483 * still attempt to free the resource. Prevent a NULL pointer access. 5484 */ 5485 if (base != NULL) { 5486 for (x = 0; x < sli4->config.extent[rtype].number; x++) { 5487 if ((rid >= base[x]) && (rid < (base[x] + size))) { 5488 rid -= base[x]; 5489 ocs_bitmap_clear(sli4->config.extent[rtype].use_map, 5490 (x * size) + rid); 5491 rc = 0; 5492 break; 5493 } 5494 } 5495 } 5496 break; 5497 default: 5498 ; 5499 } 5500 5501 return rc; 5502 } 5503 5504 int32_t 5505 sli_resource_reset(sli4_t *sli4, sli4_resource_e rtype) 5506 { 5507 int32_t rc = -1; 5508 uint32_t i; 5509 5510 switch (rtype) { 5511 case SLI_RSRC_FCOE_VFI: 5512 case SLI_RSRC_FCOE_VPI: 5513 case SLI_RSRC_FCOE_RPI: 5514 case SLI_RSRC_FCOE_XRI: 5515 for (i = 0; i < sli4->config.extent[rtype].map_size; i++) { 5516 ocs_bitmap_clear(sli4->config.extent[rtype].use_map, i); 5517 } 5518 rc = 0; 5519 break; 5520 default: 5521 ; 5522 } 5523 5524 return rc; 5525 } 5526 5527 /** 5528 * @ingroup sli 5529 * @brief Parse an EQ entry to retrieve the CQ_ID for this event. 5530 * 5531 * @param sli4 SLI context. 5532 * @param buf Pointer to the EQ entry. 5533 * @param cq_id CQ_ID for this entry (only valid on success). 5534 * 5535 * @return 5536 * - 0 if success. 5537 * - < 0 if error. 5538 * - > 0 if firmware detects EQ overflow. 5539 */ 5540 int32_t 5541 sli_eq_parse(sli4_t *sli4, uint8_t *buf, uint16_t *cq_id) 5542 { 5543 sli4_eqe_t *eqe = (void *)buf; 5544 int32_t rc = 0; 5545 5546 if (!sli4 || !buf || !cq_id) { 5547 ocs_log_err(NULL, "bad parameters sli4=%p buf=%p cq_id=%p\n", 5548 sli4, buf, cq_id); 5549 return -1; 5550 } 5551 5552 switch (eqe->major_code) { 5553 case SLI4_MAJOR_CODE_STANDARD: 5554 *cq_id = eqe->resource_id; 5555 break; 5556 case SLI4_MAJOR_CODE_SENTINEL: 5557 ocs_log_debug(sli4->os, "sentinel EQE\n"); 5558 rc = 1; 5559 break; 5560 default: 5561 ocs_log_test(sli4->os, "Unsupported EQE: major %x minor %x\n", 5562 eqe->major_code, eqe->minor_code); 5563 rc = -1; 5564 } 5565 5566 return rc; 5567 } 5568 5569 /** 5570 * @ingroup sli 5571 * @brief Parse a CQ entry to retrieve the event type and the associated queue. 5572 * 5573 * @param sli4 SLI context. 5574 * @param cq CQ to process. 5575 * @param cqe Pointer to the CQ entry. 5576 * @param etype CQ event type. 5577 * @param q_id Queue ID associated with this completion message 5578 * (that is, MQ_ID, RQ_ID, and so on). 5579 * 5580 * @return 5581 * - 0 if call completed correctly and CQE status is SUCCESS. 5582 * - -1 if call failed (no CQE status). 5583 * - Other value if call completed correctly and return value is a CQE status value. 5584 */ 5585 int32_t 5586 sli_cq_parse(sli4_t *sli4, sli4_queue_t *cq, uint8_t *cqe, sli4_qentry_e *etype, 5587 uint16_t *q_id) 5588 { 5589 int32_t rc = 0; 5590 5591 if (!sli4 || !cq || !cqe || !etype) { 5592 ocs_log_err(NULL, "bad parameters sli4=%p cq=%p cqe=%p etype=%p q_id=%p\n", 5593 sli4, cq, cqe, etype, q_id); 5594 return -1; 5595 } 5596 5597 if (cq->u.flag.is_mq) { 5598 sli4_mcqe_t *mcqe = (void *)cqe; 5599 5600 if (mcqe->ae) { 5601 *etype = SLI_QENTRY_ASYNC; 5602 } else { 5603 *etype = SLI_QENTRY_MQ; 5604 rc = sli_cqe_mq(mcqe); 5605 } 5606 *q_id = -1; 5607 } else if (SLI4_PORT_TYPE_FC == sli4->port_type) { 5608 rc = sli_fc_cqe_parse(sli4, cq, cqe, etype, q_id); 5609 } else { 5610 ocs_log_test(sli4->os, "implement CQE parsing type = %#x\n", 5611 sli4->port_type); 5612 rc = -1; 5613 } 5614 5615 return rc; 5616 } 5617 5618 /** 5619 * @ingroup sli 5620 * @brief Cause chip to enter an unrecoverable error state. 5621 * 5622 * @par Description 5623 * Cause chip to enter an unrecoverable error state. This is 5624 * used when detecting unexpected FW behavior so FW can be 5625 * hwted from the driver as soon as error is detected. 5626 * 5627 * @param sli4 SLI context. 5628 * @param dump Generate dump as part of reset. 5629 * 5630 * @return Returns 0 if call completed correctly, or -1 if call failed (unsupported chip). 5631 */ 5632 int32_t sli_raise_ue(sli4_t *sli4, uint8_t dump) 5633 { 5634 #define FDD 2 5635 if (SLI4_IF_TYPE_BE3_SKH_PF == sli_get_if_type(sli4)) { 5636 switch(sli_get_asic_type(sli4)) { 5637 case SLI4_ASIC_TYPE_BE3: { 5638 sli_reg_write(sli4, SLI4_REG_SW_UE_CSR1, 0xffffffff); 5639 sli_reg_write(sli4, SLI4_REG_SW_UE_CSR2, 0); 5640 break; 5641 } 5642 case SLI4_ASIC_TYPE_SKYHAWK: { 5643 uint32_t value; 5644 value = ocs_config_read32(sli4->os, SLI4_SW_UE_REG); 5645 ocs_config_write32(sli4->os, SLI4_SW_UE_REG, (value | (1U << 24))); 5646 break; 5647 } 5648 default: 5649 ocs_log_test(sli4->os, "invalid asic type %d\n", sli_get_asic_type(sli4)); 5650 return -1; 5651 } 5652 } else if ((SLI4_IF_TYPE_LANCER_FC_ETH == sli_get_if_type(sli4)) || 5653 (SLI4_IF_TYPE_LANCER_G7 == sli_get_if_type(sli4))) { 5654 if (FDD == dump) { 5655 sli_reg_write(sli4, SLI4_REG_SLIPORT_CONTROL, SLI4_SLIPORT_CONTROL_FDD | SLI4_SLIPORT_CONTROL_IP); 5656 } else { 5657 uint32_t value = SLI4_PHYDEV_CONTROL_FRST; 5658 if (dump == 1) { 5659 value |= SLI4_PHYDEV_CONTROL_DD; 5660 } 5661 sli_reg_write(sli4, SLI4_REG_PHYSDEV_CONTROL, value); 5662 } 5663 } else { 5664 ocs_log_test(sli4->os, "invalid iftype=%d\n", sli_get_if_type(sli4)); 5665 return -1; 5666 } 5667 return 0; 5668 } 5669 5670 /** 5671 * @ingroup sli 5672 * @brief Read the SLIPORT_STATUS register to check if a dump is present. 5673 * 5674 * @param sli4 SLI context. 5675 * 5676 * @return Returns 1 if the chip is ready, or 0 if the chip is not ready, 2 if fdp is present. 5677 */ 5678 int32_t sli_dump_is_ready(sli4_t *sli4) 5679 { 5680 int32_t rc = 0; 5681 uint32_t port_val; 5682 uint32_t bmbx_val; 5683 uint32_t uerr_lo; 5684 uint32_t uerr_hi; 5685 uint32_t uerr_mask_lo; 5686 uint32_t uerr_mask_hi; 5687 5688 if (SLI4_IF_TYPE_BE3_SKH_PF == sli_get_if_type(sli4)) { 5689 /* for iftype=0, dump ready when UE is encountered */ 5690 uerr_lo = sli_reg_read(sli4, SLI4_REG_UERR_STATUS_LO); 5691 uerr_hi = sli_reg_read(sli4, SLI4_REG_UERR_STATUS_HI); 5692 uerr_mask_lo = sli_reg_read(sli4, SLI4_REG_UERR_MASK_LO); 5693 uerr_mask_hi = sli_reg_read(sli4, SLI4_REG_UERR_MASK_HI); 5694 if ((uerr_lo & ~uerr_mask_lo) || (uerr_hi & ~uerr_mask_hi)) { 5695 rc = 1; 5696 } 5697 5698 } else if ((SLI4_IF_TYPE_LANCER_FC_ETH == sli_get_if_type(sli4)) || 5699 (SLI4_IF_TYPE_LANCER_G7 == sli_get_if_type(sli4))) { 5700 /* 5701 * Ensure that the port is ready AND the mailbox is 5702 * ready before signaling that the dump is ready to go. 5703 */ 5704 port_val = sli_reg_read(sli4, SLI4_REG_SLIPORT_STATUS); 5705 bmbx_val = sli_reg_read(sli4, SLI4_REG_BMBX); 5706 5707 if ((bmbx_val & SLI4_BMBX_RDY) && 5708 SLI4_PORT_STATUS_READY(port_val)) { 5709 if(SLI4_PORT_STATUS_DUMP_PRESENT(port_val)) { 5710 rc = 1; 5711 }else if( SLI4_PORT_STATUS_FDP_PRESENT(port_val)) { 5712 rc = 2; 5713 } 5714 } 5715 } else { 5716 ocs_log_test(sli4->os, "invalid iftype=%d\n", sli_get_if_type(sli4)); 5717 return -1; 5718 } 5719 return rc; 5720 } 5721 5722 /** 5723 * @ingroup sli 5724 * @brief Read the SLIPORT_STATUS register to check if a dump is present. 5725 * 5726 * @param sli4 SLI context. 5727 * 5728 * @return 5729 * - 0 if call completed correctly and no dump is present. 5730 * - 1 if call completed and dump is present. 5731 * - -1 if call failed (unsupported chip). 5732 */ 5733 int32_t sli_dump_is_present(sli4_t *sli4) 5734 { 5735 uint32_t val; 5736 uint32_t ready; 5737 5738 if ((SLI4_IF_TYPE_LANCER_FC_ETH != sli_get_if_type(sli4)) && 5739 (SLI4_IF_TYPE_LANCER_G7 != sli_get_if_type(sli4))) { 5740 ocs_log_test(sli4->os, "Function only supported for I/F type 2"); 5741 return -1; 5742 } 5743 5744 /* If the chip is not ready, then there cannot be a dump */ 5745 ready = sli_wait_for_fw_ready(sli4, SLI4_INIT_PORT_DELAY_US); 5746 if (!ready) { 5747 return 0; 5748 } 5749 5750 val = sli_reg_read(sli4, SLI4_REG_SLIPORT_STATUS); 5751 if (UINT32_MAX == val) { 5752 ocs_log_err(sli4->os, "error reading SLIPORT_STATUS\n"); 5753 return -1; 5754 } else { 5755 return ((val & SLI4_PORT_STATUS_DIP) ? 1 : 0); 5756 } 5757 } 5758 5759 /** 5760 * @ingroup sli 5761 * @brief Read the SLIPORT_STATUS register to check if the reset required is set. 5762 * 5763 * @param sli4 SLI context. 5764 * 5765 * @return 5766 * - 0 if call completed correctly and reset is not required. 5767 * - 1 if call completed and reset is required. 5768 * - -1 if call failed. 5769 */ 5770 int32_t sli_reset_required(sli4_t *sli4) 5771 { 5772 uint32_t val; 5773 5774 if (SLI4_IF_TYPE_BE3_SKH_PF == sli_get_if_type(sli4)) { 5775 ocs_log_test(sli4->os, "reset required N/A for iftype 0\n"); 5776 return 0; 5777 } 5778 5779 val = sli_reg_read(sli4, SLI4_REG_SLIPORT_STATUS); 5780 if (UINT32_MAX == val) { 5781 ocs_log_err(sli4->os, "error reading SLIPORT_STATUS\n"); 5782 return -1; 5783 } else { 5784 return ((val & SLI4_PORT_STATUS_RN) ? 1 : 0); 5785 } 5786 } 5787 5788 /** 5789 * @ingroup sli 5790 * @brief Read the SLIPORT_SEMAPHORE and SLIPORT_STATUS registers to check if 5791 * the port status indicates that a FW error has occurred. 5792 * 5793 * @param sli4 SLI context. 5794 * 5795 * @return 5796 * - 0 if call completed correctly and no FW error occurred. 5797 * - > 0 which indicates that a FW error has occurred. 5798 * - -1 if call failed. 5799 */ 5800 int32_t sli_fw_error_status(sli4_t *sli4) 5801 { 5802 uint32_t sliport_semaphore; 5803 int32_t rc = 0; 5804 5805 sliport_semaphore = sli_reg_read(sli4, SLI4_REG_SLIPORT_SEMAPHORE); 5806 if (UINT32_MAX == sliport_semaphore) { 5807 ocs_log_err(sli4->os, "error reading SLIPORT_SEMAPHORE register\n"); 5808 return 1; 5809 } 5810 rc = (SLI4_PORT_SEMAPHORE_IN_ERR(sliport_semaphore) ? 1 : 0); 5811 5812 if (rc == 0) { 5813 if (SLI4_IF_TYPE_BE3_SKH_PF == sli4->if_type || 5814 (SLI4_IF_TYPE_BE3_SKH_VF == sli4->if_type)) { 5815 uint32_t uerr_mask_lo, uerr_mask_hi; 5816 uint32_t uerr_status_lo, uerr_status_hi; 5817 5818 uerr_mask_lo = sli_reg_read(sli4, SLI4_REG_UERR_MASK_LO); 5819 uerr_mask_hi = sli_reg_read(sli4, SLI4_REG_UERR_MASK_HI); 5820 uerr_status_lo = sli_reg_read(sli4, SLI4_REG_UERR_STATUS_LO); 5821 uerr_status_hi = sli_reg_read(sli4, SLI4_REG_UERR_STATUS_HI); 5822 if ((uerr_mask_lo & uerr_status_lo) != 0 || 5823 (uerr_mask_hi & uerr_status_hi) != 0) { 5824 rc = 1; 5825 } 5826 } else if (SLI4_IF_TYPE_LANCER_FC_ETH == sli4->if_type || 5827 SLI4_IF_TYPE_LANCER_G7 == sli4->if_type) { 5828 uint32_t sliport_status; 5829 5830 sliport_status = sli_reg_read(sli4, SLI4_REG_SLIPORT_STATUS); 5831 rc = (SLI4_PORT_STATUS_ERROR(sliport_status) ? 1 : 0); 5832 } 5833 } 5834 return rc; 5835 } 5836 5837 /** 5838 * @ingroup sli 5839 * @brief Determine if the chip FW is in a ready state 5840 * 5841 * @param sli4 SLI context. 5842 * 5843 * @return 5844 * - 0 if call completed correctly and FW is not ready. 5845 * - 1 if call completed correctly and FW is ready. 5846 * - -1 if call failed. 5847 */ 5848 int32_t 5849 sli_fw_ready(sli4_t *sli4) 5850 { 5851 uint32_t val; 5852 int32_t rc = -1; 5853 5854 /* 5855 * Is firmware ready for operation? Check needed depends on IF_TYPE 5856 */ 5857 if (SLI4_IF_TYPE_BE3_SKH_PF == sli4->if_type || 5858 SLI4_IF_TYPE_BE3_SKH_VF == sli4->if_type) { 5859 val = sli_reg_read(sli4, SLI4_REG_SLIPORT_SEMAPHORE); 5860 rc = ((SLI4_PORT_SEMAPHORE_STATUS_POST_READY == 5861 SLI4_PORT_SEMAPHORE_PORT(val)) && 5862 (!SLI4_PORT_SEMAPHORE_IN_ERR(val)) ? 1 : 0); 5863 } else if (SLI4_IF_TYPE_LANCER_FC_ETH == sli4->if_type || 5864 SLI4_IF_TYPE_LANCER_G7 == sli4->if_type) { 5865 val = sli_reg_read(sli4, SLI4_REG_SLIPORT_STATUS); 5866 rc = (SLI4_PORT_STATUS_READY(val) ? 1 : 0); 5867 } 5868 return rc; 5869 } 5870 5871 /** 5872 * @ingroup sli 5873 * @brief Determine if the link can be configured 5874 * 5875 * @param sli4 SLI context. 5876 * 5877 * @return 5878 * - 0 if link is not configurable. 5879 * - 1 if link is configurable. 5880 */ 5881 int32_t sli_link_is_configurable(sli4_t *sli) 5882 { 5883 int32_t rc = 0; 5884 /* 5885 * Link config works on: Skyhawk and Lancer 5886 * Link config does not work on: LancerG6 5887 */ 5888 5889 switch (sli_get_asic_type(sli)) { 5890 case SLI4_ASIC_TYPE_SKYHAWK: 5891 case SLI4_ASIC_TYPE_LANCER: 5892 case SLI4_ASIC_TYPE_CORSAIR: 5893 rc = 1; 5894 break; 5895 case SLI4_ASIC_TYPE_LANCERG6: 5896 case SLI4_ASIC_TYPE_LANCERG7: 5897 case SLI4_ASIC_TYPE_BE3: 5898 default: 5899 rc = 0; 5900 break; 5901 } 5902 5903 return rc; 5904 5905 } 5906 5907 /* vim: set noexpandtab textwidth=120: */ 5908 5909 /** 5910 * @ingroup sli_fc 5911 * @brief Write an FCOE_WQ_CREATE command. 5912 * 5913 * @param sli4 SLI context. 5914 * @param buf Destination buffer for the command. 5915 * @param size Buffer size, in bytes. 5916 * @param qmem DMA memory for the queue. 5917 * @param cq_id Associated CQ_ID. 5918 * @param ulp The ULP to bind 5919 * 5920 * @note This creates a Version 0 message. 5921 * 5922 * @return Returns the number of bytes written. 5923 */ 5924 int32_t 5925 sli_cmd_fcoe_wq_create(sli4_t *sli4, void *buf, size_t size, 5926 ocs_dma_t *qmem, uint16_t cq_id, uint16_t ulp) 5927 { 5928 sli4_req_fcoe_wq_create_t *wq = NULL; 5929 uint32_t sli_config_off = 0; 5930 uint32_t p; 5931 uintptr_t addr; 5932 5933 if (SLI4_PORT_TYPE_FC == sli4->port_type) { 5934 uint32_t payload_size; 5935 5936 /* Payload length must accommodate both request and response */ 5937 payload_size = max(sizeof(sli4_req_fcoe_wq_create_t), 5938 sizeof(sli4_res_common_create_queue_t)); 5939 5940 sli_config_off = sli_cmd_sli_config(sli4, buf, size, payload_size, 5941 NULL); 5942 } 5943 wq = (sli4_req_fcoe_wq_create_t *)((uint8_t *)buf + sli_config_off); 5944 5945 wq->hdr.opcode = SLI4_OPC_FCOE_WQ_CREATE; 5946 wq->hdr.subsystem = SLI4_SUBSYSTEM_FCFCOE; 5947 wq->hdr.request_length = sizeof(sli4_req_fcoe_wq_create_t) - 5948 sizeof(sli4_req_hdr_t); 5949 /* valid values for number of pages: 1-4 (sec 4.5.1) */ 5950 wq->num_pages = sli_page_count(qmem->size, SLI_PAGE_SIZE); 5951 if (!wq->num_pages || (wq->num_pages > SLI4_FCOE_WQ_CREATE_V0_MAX_PAGES)) { 5952 return 0; 5953 } 5954 5955 wq->cq_id = cq_id; 5956 5957 if (sli4->config.dual_ulp_capable) { 5958 wq->dua = 1; 5959 wq->bqu = 1; 5960 wq->ulp = ulp; 5961 } 5962 5963 for (p = 0, addr = qmem->phys; 5964 p < wq->num_pages; 5965 p++, addr += SLI_PAGE_SIZE) { 5966 wq->page_physical_address[p].low = ocs_addr32_lo(addr); 5967 wq->page_physical_address[p].high = ocs_addr32_hi(addr); 5968 } 5969 5970 return(sli_config_off + sizeof(sli4_req_fcoe_wq_create_t)); 5971 } 5972 5973 /** 5974 * @ingroup sli_fc 5975 * @brief Write an FCOE_WQ_CREATE_V1 command. 5976 * 5977 * @param sli4 SLI context. 5978 * @param buf Destination buffer for the command. 5979 * @param size Buffer size, in bytes. 5980 * @param qmem DMA memory for the queue. 5981 * @param cq_id Associated CQ_ID. 5982 * @param ignored This parameter carries the ULP for WQ (ignored for V1) 5983 5984 * 5985 * @return Returns the number of bytes written. 5986 */ 5987 int32_t 5988 sli_cmd_fcoe_wq_create_v1(sli4_t *sli4, void *buf, size_t size, ocs_dma_t *qmem, 5989 uint16_t cq_id, uint16_t ignored) 5990 { 5991 sli4_req_fcoe_wq_create_v1_t *wq = NULL; 5992 uint32_t sli_config_off = 0; 5993 uint32_t p; 5994 uintptr_t addr; 5995 uint32_t page_size = 0; 5996 uint32_t page_bytes = 0; 5997 uint32_t n_wqe = 0; 5998 5999 if (SLI4_PORT_TYPE_FC == sli4->port_type) { 6000 uint32_t payload_size; 6001 6002 /* Payload length must accommodate both request and response */ 6003 payload_size = max(sizeof(sli4_req_fcoe_wq_create_v1_t), 6004 sizeof(sli4_res_common_create_queue_t)); 6005 6006 sli_config_off = sli_cmd_sli_config(sli4, buf, size, payload_size, 6007 NULL); 6008 } 6009 wq = (sli4_req_fcoe_wq_create_v1_t *)((uint8_t *)buf + sli_config_off); 6010 6011 wq->hdr.opcode = SLI4_OPC_FCOE_WQ_CREATE; 6012 wq->hdr.subsystem = SLI4_SUBSYSTEM_FCFCOE; 6013 wq->hdr.request_length = sizeof(sli4_req_fcoe_wq_create_v1_t) - 6014 sizeof(sli4_req_hdr_t); 6015 wq->hdr.version = 1; 6016 6017 n_wqe = qmem->size / sli4->config.wqe_size; 6018 6019 /* This heuristic to determine the page size is simplistic 6020 * but could be made more sophisticated 6021 */ 6022 switch (qmem->size) { 6023 case 4096: 6024 case 8192: 6025 case 16384: 6026 case 32768: 6027 page_size = 1; 6028 break; 6029 case 65536: 6030 page_size = 2; 6031 break; 6032 case 131072: 6033 page_size = 4; 6034 break; 6035 case 262144: 6036 page_size = 8; 6037 break; 6038 case 524288: 6039 page_size = 10; 6040 break; 6041 default: 6042 return 0; 6043 } 6044 page_bytes = page_size * SLI_PAGE_SIZE; 6045 6046 /* valid values for number of pages: 1-8 */ 6047 wq->num_pages = sli_page_count(qmem->size, page_bytes); 6048 if (!wq->num_pages || (wq->num_pages > SLI4_FCOE_WQ_CREATE_V1_MAX_PAGES)) { 6049 return 0; 6050 } 6051 6052 wq->cq_id = cq_id; 6053 6054 wq->page_size = page_size; 6055 6056 if (sli4->config.wqe_size == SLI4_WQE_EXT_BYTES) { 6057 wq->wqe_size = SLI4_WQE_EXT_SIZE; 6058 } else { 6059 wq->wqe_size = SLI4_WQE_SIZE; 6060 } 6061 6062 wq->wqe_count = n_wqe; 6063 6064 for (p = 0, addr = qmem->phys; 6065 p < wq->num_pages; 6066 p++, addr += page_bytes) { 6067 wq->page_physical_address[p].low = ocs_addr32_lo(addr); 6068 wq->page_physical_address[p].high = ocs_addr32_hi(addr); 6069 } 6070 6071 return(sli_config_off + sizeof(sli4_req_fcoe_wq_create_v1_t)); 6072 } 6073 6074 /** 6075 * @ingroup sli_fc 6076 * @brief Write an FCOE_WQ_DESTROY command. 6077 * 6078 * @param sli4 SLI context. 6079 * @param buf Destination buffer for the command. 6080 * @param size Buffer size, in bytes. 6081 * @param wq_id WQ_ID. 6082 * 6083 * @return Returns the number of bytes written. 6084 */ 6085 int32_t 6086 sli_cmd_fcoe_wq_destroy(sli4_t *sli4, void *buf, size_t size, uint16_t wq_id) 6087 { 6088 sli4_req_fcoe_wq_destroy_t *wq = NULL; 6089 uint32_t sli_config_off = 0; 6090 6091 if (SLI4_PORT_TYPE_FC == sli4->port_type) { 6092 uint32_t payload_size; 6093 6094 /* Payload length must accommodate both request and response */ 6095 payload_size = max(sizeof(sli4_req_fcoe_wq_destroy_t), 6096 sizeof(sli4_res_hdr_t)); 6097 6098 sli_config_off = sli_cmd_sli_config(sli4, buf, size, payload_size, 6099 NULL); 6100 } 6101 wq = (sli4_req_fcoe_wq_destroy_t *)((uint8_t *)buf + sli_config_off); 6102 6103 wq->hdr.opcode = SLI4_OPC_FCOE_WQ_DESTROY; 6104 wq->hdr.subsystem = SLI4_SUBSYSTEM_FCFCOE; 6105 wq->hdr.request_length = sizeof(sli4_req_fcoe_wq_destroy_t) - 6106 sizeof(sli4_req_hdr_t); 6107 6108 wq->wq_id = wq_id; 6109 6110 return(sli_config_off + sizeof(sli4_req_fcoe_wq_destroy_t)); 6111 } 6112 6113 /** 6114 * @ingroup sli_fc 6115 * @brief Write an FCOE_POST_SGL_PAGES command. 6116 * 6117 * @param sli4 SLI context. 6118 * @param buf Destination buffer for the command. 6119 * @param size Buffer size, in bytes. 6120 * @param xri starting XRI 6121 * @param xri_count XRI 6122 * @param page0 First SGL memory page. 6123 * @param page1 Second SGL memory page (optional). 6124 * @param dma DMA buffer for non-embedded mailbox command (options) 6125 * 6126 * if non-embedded mbx command is used, dma buffer must be at least (32 + xri_count*16) in length 6127 * 6128 * @return Returns the number of bytes written. 6129 */ 6130 int32_t 6131 sli_cmd_fcoe_post_sgl_pages(sli4_t *sli4, void *buf, size_t size, 6132 uint16_t xri, uint32_t xri_count, ocs_dma_t *page0[], ocs_dma_t *page1[], ocs_dma_t *dma) 6133 { 6134 sli4_req_fcoe_post_sgl_pages_t *post = NULL; 6135 uint32_t sli_config_off = 0; 6136 uint32_t i; 6137 6138 if (SLI4_PORT_TYPE_FC == sli4->port_type) { 6139 uint32_t payload_size; 6140 6141 /* Payload length must accommodate both request and response */ 6142 payload_size = max(sizeof(sli4_req_fcoe_post_sgl_pages_t), 6143 sizeof(sli4_res_hdr_t)); 6144 6145 sli_config_off = sli_cmd_sli_config(sli4, buf, size, payload_size, 6146 dma); 6147 } 6148 if (dma) { 6149 post = dma->virt; 6150 ocs_memset(post, 0, dma->size); 6151 } else { 6152 post = (sli4_req_fcoe_post_sgl_pages_t *)((uint8_t *)buf + sli_config_off); 6153 } 6154 6155 post->hdr.opcode = SLI4_OPC_FCOE_POST_SGL_PAGES; 6156 post->hdr.subsystem = SLI4_SUBSYSTEM_FCFCOE; 6157 /* payload size calculation 6158 * 4 = xri_start + xri_count 6159 * xri_count = # of XRI's registered 6160 * sizeof(uint64_t) = physical address size 6161 * 2 = # of physical addresses per page set 6162 */ 6163 post->hdr.request_length = 4 + (xri_count * (sizeof(uint64_t) * 2)); 6164 6165 post->xri_start = xri; 6166 post->xri_count = xri_count; 6167 6168 for (i = 0; i < xri_count; i++) { 6169 post->page_set[i].page0_low = ocs_addr32_lo(page0[i]->phys); 6170 post->page_set[i].page0_high = ocs_addr32_hi(page0[i]->phys); 6171 } 6172 6173 if (page1) { 6174 for (i = 0; i < xri_count; i++) { 6175 post->page_set[i].page1_low = ocs_addr32_lo(page1[i]->phys); 6176 post->page_set[i].page1_high = ocs_addr32_hi(page1[i]->phys); 6177 } 6178 } 6179 6180 return dma ? sli_config_off : (sli_config_off + sizeof(sli4_req_fcoe_post_sgl_pages_t)); 6181 } 6182 6183 /** 6184 * @ingroup sli_fc 6185 * @brief Write an FCOE_RQ_CREATE command. 6186 * 6187 * @param sli4 SLI context. 6188 * @param buf Destination buffer for the command. 6189 * @param size Buffer size, in bytes. 6190 * @param qmem DMA memory for the queue. 6191 * @param cq_id Associated CQ_ID. 6192 * @param ulp This parameter carries the ULP for the RQ 6193 * @param buffer_size Buffer size pointed to by each RQE. 6194 * 6195 * @note This creates a Version 0 message. 6196 * 6197 * @return Returns the number of bytes written. 6198 */ 6199 int32_t 6200 sli_cmd_fcoe_rq_create(sli4_t *sli4, void *buf, size_t size, 6201 ocs_dma_t *qmem, uint16_t cq_id, uint16_t ulp, uint16_t buffer_size) 6202 { 6203 sli4_req_fcoe_rq_create_t *rq = NULL; 6204 uint32_t sli_config_off = 0; 6205 uint32_t p; 6206 uintptr_t addr; 6207 6208 if (SLI4_PORT_TYPE_FC == sli4->port_type) { 6209 uint32_t payload_size; 6210 6211 /* Payload length must accommodate both request and response */ 6212 payload_size = max(sizeof(sli4_req_fcoe_rq_create_t), 6213 sizeof(sli4_res_common_create_queue_t)); 6214 6215 sli_config_off = sli_cmd_sli_config(sli4, buf, size, payload_size, 6216 NULL); 6217 } 6218 rq = (sli4_req_fcoe_rq_create_t *)((uint8_t *)buf + sli_config_off); 6219 6220 rq->hdr.opcode = SLI4_OPC_FCOE_RQ_CREATE; 6221 rq->hdr.subsystem = SLI4_SUBSYSTEM_FCFCOE; 6222 rq->hdr.request_length = sizeof(sli4_req_fcoe_rq_create_t) - 6223 sizeof(sli4_req_hdr_t); 6224 /* valid values for number of pages: 1-8 (sec 4.5.6) */ 6225 rq->num_pages = sli_page_count(qmem->size, SLI_PAGE_SIZE); 6226 if (!rq->num_pages || (rq->num_pages > SLI4_FCOE_RQ_CREATE_V0_MAX_PAGES)) { 6227 ocs_log_test(sli4->os, "num_pages %d not valid\n", rq->num_pages); 6228 return 0; 6229 } 6230 6231 /* 6232 * RQE count is the log base 2 of the total number of entries 6233 */ 6234 rq->rqe_count = ocs_lg2(qmem->size / SLI4_FCOE_RQE_SIZE); 6235 6236 if ((buffer_size < SLI4_FCOE_RQ_CREATE_V0_MIN_BUF_SIZE) || 6237 (buffer_size > SLI4_FCOE_RQ_CREATE_V0_MAX_BUF_SIZE)) { 6238 ocs_log_err(sli4->os, "buffer_size %d out of range (%d-%d)\n", 6239 buffer_size, 6240 SLI4_FCOE_RQ_CREATE_V0_MIN_BUF_SIZE, 6241 SLI4_FCOE_RQ_CREATE_V0_MAX_BUF_SIZE); 6242 return -1; 6243 } 6244 rq->buffer_size = buffer_size; 6245 6246 rq->cq_id = cq_id; 6247 6248 if (sli4->config.dual_ulp_capable) { 6249 rq->dua = 1; 6250 rq->bqu = 1; 6251 rq->ulp = ulp; 6252 } 6253 6254 for (p = 0, addr = qmem->phys; 6255 p < rq->num_pages; 6256 p++, addr += SLI_PAGE_SIZE) { 6257 rq->page_physical_address[p].low = ocs_addr32_lo(addr); 6258 rq->page_physical_address[p].high = ocs_addr32_hi(addr); 6259 } 6260 6261 return(sli_config_off + sizeof(sli4_req_fcoe_rq_create_t)); 6262 } 6263 6264 /** 6265 * @ingroup sli_fc 6266 * @brief Write an FCOE_RQ_CREATE_V1 command. 6267 * 6268 * @param sli4 SLI context. 6269 * @param buf Destination buffer for the command. 6270 * @param size Buffer size, in bytes. 6271 * @param qmem DMA memory for the queue. 6272 * @param cq_id Associated CQ_ID. 6273 * @param ulp This parameter carries the ULP for RQ (ignored for V1) 6274 * @param buffer_size Buffer size pointed to by each RQE. 6275 * 6276 * @note This creates a Version 0 message 6277 * 6278 * @return Returns the number of bytes written. 6279 */ 6280 int32_t 6281 sli_cmd_fcoe_rq_create_v1(sli4_t *sli4, void *buf, size_t size, 6282 ocs_dma_t *qmem, uint16_t cq_id, uint16_t ulp, 6283 uint16_t buffer_size) 6284 { 6285 sli4_req_fcoe_rq_create_v1_t *rq = NULL; 6286 uint32_t sli_config_off = 0; 6287 uint32_t p; 6288 uintptr_t addr; 6289 6290 if (SLI4_PORT_TYPE_FC == sli4->port_type) { 6291 uint32_t payload_size; 6292 6293 /* Payload length must accommodate both request and response */ 6294 payload_size = max(sizeof(sli4_req_fcoe_rq_create_v1_t), 6295 sizeof(sli4_res_common_create_queue_t)); 6296 6297 sli_config_off = sli_cmd_sli_config(sli4, buf, size, payload_size, 6298 NULL); 6299 } 6300 rq = (sli4_req_fcoe_rq_create_v1_t *)((uint8_t *)buf + sli_config_off); 6301 6302 rq->hdr.opcode = SLI4_OPC_FCOE_RQ_CREATE; 6303 rq->hdr.subsystem = SLI4_SUBSYSTEM_FCFCOE; 6304 rq->hdr.request_length = sizeof(sli4_req_fcoe_rq_create_v1_t) - 6305 sizeof(sli4_req_hdr_t); 6306 rq->hdr.version = 1; 6307 6308 /* Disable "no buffer warnings" to avoid Lancer bug */ 6309 rq->dnb = TRUE; 6310 6311 /* valid values for number of pages: 1-8 (sec 4.5.6) */ 6312 rq->num_pages = sli_page_count(qmem->size, SLI_PAGE_SIZE); 6313 if (!rq->num_pages || (rq->num_pages > SLI4_FCOE_RQ_CREATE_V1_MAX_PAGES)) { 6314 ocs_log_test(sli4->os, "num_pages %d not valid, max %d\n", 6315 rq->num_pages, SLI4_FCOE_RQ_CREATE_V1_MAX_PAGES); 6316 return 0; 6317 } 6318 6319 /* 6320 * RQE count is the total number of entries (note not lg2(# entries)) 6321 */ 6322 rq->rqe_count = qmem->size / SLI4_FCOE_RQE_SIZE; 6323 6324 rq->rqe_size = SLI4_FCOE_RQE_SIZE_8; 6325 6326 rq->page_size = SLI4_FCOE_RQ_PAGE_SIZE_4096; 6327 6328 if ((buffer_size < sli4->config.rq_min_buf_size) || 6329 (buffer_size > sli4->config.rq_max_buf_size)) { 6330 ocs_log_err(sli4->os, "buffer_size %d out of range (%d-%d)\n", 6331 buffer_size, 6332 sli4->config.rq_min_buf_size, 6333 sli4->config.rq_max_buf_size); 6334 return -1; 6335 } 6336 rq->buffer_size = buffer_size; 6337 6338 rq->cq_id = cq_id; 6339 6340 for (p = 0, addr = qmem->phys; 6341 p < rq->num_pages; 6342 p++, addr += SLI_PAGE_SIZE) { 6343 rq->page_physical_address[p].low = ocs_addr32_lo(addr); 6344 rq->page_physical_address[p].high = ocs_addr32_hi(addr); 6345 } 6346 6347 return(sli_config_off + sizeof(sli4_req_fcoe_rq_create_v1_t)); 6348 } 6349 6350 /** 6351 * @ingroup sli_fc 6352 * @brief Write an FCOE_RQ_DESTROY command. 6353 * 6354 * @param sli4 SLI context. 6355 * @param buf Destination buffer for the command. 6356 * @param size Buffer size, in bytes. 6357 * @param rq_id RQ_ID. 6358 * 6359 * @return Returns the number of bytes written. 6360 */ 6361 int32_t 6362 sli_cmd_fcoe_rq_destroy(sli4_t *sli4, void *buf, size_t size, uint16_t rq_id) 6363 { 6364 sli4_req_fcoe_rq_destroy_t *rq = NULL; 6365 uint32_t sli_config_off = 0; 6366 6367 if (SLI4_PORT_TYPE_FC == sli4->port_type) { 6368 uint32_t payload_size; 6369 6370 /* Payload length must accommodate both request and response */ 6371 payload_size = max(sizeof(sli4_req_fcoe_rq_destroy_t), 6372 sizeof(sli4_res_hdr_t)); 6373 6374 sli_config_off = sli_cmd_sli_config(sli4, buf, size, payload_size, 6375 NULL); 6376 } 6377 rq = (sli4_req_fcoe_rq_destroy_t *)((uint8_t *)buf + sli_config_off); 6378 6379 rq->hdr.opcode = SLI4_OPC_FCOE_RQ_DESTROY; 6380 rq->hdr.subsystem = SLI4_SUBSYSTEM_FCFCOE; 6381 rq->hdr.request_length = sizeof(sli4_req_fcoe_rq_destroy_t) - 6382 sizeof(sli4_req_hdr_t); 6383 6384 rq->rq_id = rq_id; 6385 6386 return(sli_config_off + sizeof(sli4_req_fcoe_rq_destroy_t)); 6387 } 6388 6389 /** 6390 * @ingroup sli_fc 6391 * @brief Write an FCOE_READ_FCF_TABLE command. 6392 * 6393 * @note 6394 * The response of this command exceeds the size of an embedded 6395 * command and requires an external buffer with DMA capability to hold the results. 6396 * The caller should allocate the ocs_dma_t structure / memory. 6397 * 6398 * @param sli4 SLI context. 6399 * @param buf Destination buffer for the command. 6400 * @param size Buffer size, in bytes. 6401 * @param dma Pointer to DMA memory structure. This is allocated by the caller. 6402 * @param index FCF table index to retrieve. 6403 * 6404 * @return Returns the number of bytes written. 6405 */ 6406 int32_t 6407 sli_cmd_fcoe_read_fcf_table(sli4_t *sli4, void *buf, size_t size, ocs_dma_t *dma, uint16_t index) 6408 { 6409 sli4_req_fcoe_read_fcf_table_t *read_fcf = NULL; 6410 6411 if (SLI4_PORT_TYPE_FC != sli4->port_type) { 6412 ocs_log_test(sli4->os, "FCOE_READ_FCF_TABLE only supported on FC\n"); 6413 return -1; 6414 } 6415 6416 read_fcf = dma->virt; 6417 6418 ocs_memset(read_fcf, 0, sizeof(sli4_req_fcoe_read_fcf_table_t)); 6419 6420 read_fcf->hdr.opcode = SLI4_OPC_FCOE_READ_FCF_TABLE; 6421 read_fcf->hdr.subsystem = SLI4_SUBSYSTEM_FCFCOE; 6422 read_fcf->hdr.request_length = dma->size - 6423 sizeof(sli4_req_fcoe_read_fcf_table_t); 6424 read_fcf->fcf_index = index; 6425 6426 return sli_cmd_sli_config(sli4, buf, size, 0, dma); 6427 } 6428 6429 /** 6430 * @ingroup sli_fc 6431 * @brief Write an FCOE_POST_HDR_TEMPLATES command. 6432 * 6433 * @param sli4 SLI context. 6434 * @param buf Destination buffer for the command. 6435 * @param size Buffer size, in bytes. 6436 * @param dma Pointer to DMA memory structure. This is allocated by the caller. 6437 * @param rpi Starting RPI index for the header templates. 6438 * @param payload_dma Pointer to DMA memory used to hold larger descriptor counts. 6439 * 6440 * @return Returns the number of bytes written. 6441 */ 6442 int32_t 6443 sli_cmd_fcoe_post_hdr_templates(sli4_t *sli4, void *buf, size_t size, 6444 ocs_dma_t *dma, uint16_t rpi, ocs_dma_t *payload_dma) 6445 { 6446 sli4_req_fcoe_post_hdr_templates_t *template = NULL; 6447 uint32_t sli_config_off = 0; 6448 uintptr_t phys = 0; 6449 uint32_t i = 0; 6450 uint32_t page_count; 6451 uint32_t payload_size; 6452 6453 page_count = sli_page_count(dma->size, SLI_PAGE_SIZE); 6454 6455 payload_size = sizeof(sli4_req_fcoe_post_hdr_templates_t) + 6456 page_count * sizeof(sli4_physical_page_descriptor_t); 6457 6458 if (page_count > 16) { 6459 /* We can't fit more than 16 descriptors into an embedded mailbox 6460 command, it has to be non-embedded */ 6461 if (ocs_dma_alloc(sli4->os, payload_dma, payload_size, 4)) { 6462 ocs_log_err(sli4->os, "mailbox payload memory allocation fail\n"); 6463 return 0; 6464 } 6465 sli_config_off = sli_cmd_sli_config(sli4, buf, size, payload_size, payload_dma); 6466 template = (sli4_req_fcoe_post_hdr_templates_t *)payload_dma->virt; 6467 } else { 6468 sli_config_off = sli_cmd_sli_config(sli4, buf, size, payload_size, NULL); 6469 template = (sli4_req_fcoe_post_hdr_templates_t *)((uint8_t *)buf + sli_config_off); 6470 } 6471 6472 if (UINT16_MAX == rpi) { 6473 rpi = sli4->config.extent[SLI_RSRC_FCOE_RPI].base[0]; 6474 } 6475 6476 template->hdr.opcode = SLI4_OPC_FCOE_POST_HDR_TEMPLATES; 6477 template->hdr.subsystem = SLI4_SUBSYSTEM_FCFCOE; 6478 template->hdr.request_length = sizeof(sli4_req_fcoe_post_hdr_templates_t) - 6479 sizeof(sli4_req_hdr_t); 6480 6481 template->rpi_offset = rpi; 6482 template->page_count = page_count; 6483 phys = dma->phys; 6484 for (i = 0; i < template->page_count; i++) { 6485 template->page_descriptor[i].low = ocs_addr32_lo(phys); 6486 template->page_descriptor[i].high = ocs_addr32_hi(phys); 6487 6488 phys += SLI_PAGE_SIZE; 6489 } 6490 6491 return(sli_config_off + payload_size); 6492 } 6493 6494 int32_t 6495 sli_cmd_fcoe_rediscover_fcf(sli4_t *sli4, void *buf, size_t size, uint16_t index) 6496 { 6497 sli4_req_fcoe_rediscover_fcf_t *redisc = NULL; 6498 uint32_t sli_config_off = 0; 6499 6500 sli_config_off = sli_cmd_sli_config(sli4, buf, size, 6501 sizeof(sli4_req_fcoe_rediscover_fcf_t), 6502 NULL); 6503 6504 redisc = (sli4_req_fcoe_rediscover_fcf_t *)((uint8_t *)buf + sli_config_off); 6505 6506 redisc->hdr.opcode = SLI4_OPC_FCOE_REDISCOVER_FCF; 6507 redisc->hdr.subsystem = SLI4_SUBSYSTEM_FCFCOE; 6508 redisc->hdr.request_length = sizeof(sli4_req_fcoe_rediscover_fcf_t) - 6509 sizeof(sli4_req_hdr_t); 6510 6511 if (index == UINT16_MAX) { 6512 redisc->fcf_count = 0; 6513 } else { 6514 redisc->fcf_count = 1; 6515 redisc->fcf_index[0] = index; 6516 } 6517 6518 return(sli_config_off + sizeof(sli4_req_fcoe_rediscover_fcf_t)); 6519 } 6520 6521 /** 6522 * @ingroup sli_fc 6523 * @brief Write an ABORT_WQE work queue entry. 6524 * 6525 * @param sli4 SLI context. 6526 * @param buf Destination buffer for the WQE. 6527 * @param size Buffer size, in bytes. 6528 * @param type Abort type, such as XRI, abort tag, and request tag. 6529 * @param send_abts Boolean to cause the hardware to automatically generate an ABTS. 6530 * @param ids ID of IOs to abort. 6531 * @param mask Mask applied to the ID values to abort. 6532 * @param tag Tag value associated with this abort. 6533 * @param cq_id The id of the completion queue where the WQE response is sent. 6534 * @param dnrx When set to 1, this field indicates that the SLI Port must not return the associated XRI to the SLI 6535 * Port's optimized write XRI pool. 6536 * 6537 * @return Returns 0 on success, or a non-zero value on failure. 6538 */ 6539 int32_t 6540 sli_abort_wqe(sli4_t *sli4, void *buf, size_t size, sli4_abort_type_e type, uint32_t send_abts, 6541 uint32_t ids, uint32_t mask, uint16_t tag, uint16_t cq_id) 6542 { 6543 sli4_abort_wqe_t *abort = buf; 6544 6545 ocs_memset(buf, 0, size); 6546 6547 switch (type) { 6548 case SLI_ABORT_XRI: 6549 abort->criteria = SLI4_ABORT_CRITERIA_XRI_TAG; 6550 if (mask) { 6551 ocs_log_warn(sli4->os, "warning non-zero mask %#x when aborting XRI %#x\n", mask, ids); 6552 mask = 0; 6553 } 6554 break; 6555 case SLI_ABORT_ABORT_ID: 6556 abort->criteria = SLI4_ABORT_CRITERIA_ABORT_TAG; 6557 break; 6558 case SLI_ABORT_REQUEST_ID: 6559 abort->criteria = SLI4_ABORT_CRITERIA_REQUEST_TAG; 6560 break; 6561 default: 6562 ocs_log_test(sli4->os, "unsupported type %#x\n", type); 6563 return -1; 6564 } 6565 6566 abort->ia = send_abts ? 0 : 1; 6567 6568 /* Suppress ABTS retries */ 6569 abort->ir = 1; 6570 6571 abort->t_mask = mask; 6572 abort->t_tag = ids; 6573 abort->command = SLI4_WQE_ABORT; 6574 abort->request_tag = tag; 6575 abort->qosd = TRUE; 6576 abort->cq_id = cq_id; 6577 abort->cmd_type = SLI4_CMD_ABORT_WQE; 6578 6579 return 0; 6580 } 6581 6582 /** 6583 * @ingroup sli_fc 6584 * @brief Write an ELS_REQUEST64_WQE work queue entry. 6585 * 6586 * @param sli4 SLI context. 6587 * @param buf Destination buffer for the WQE. 6588 * @param size Buffer size, in bytes. 6589 * @param sgl DMA memory for the ELS request. 6590 * @param req_type ELS request type. 6591 * @param req_len Length of ELS request in bytes. 6592 * @param max_rsp_len Max length of ELS response in bytes. 6593 * @param timeout Time, in seconds, before an IO times out. Zero means 2 * R_A_TOV. 6594 * @param xri XRI for this exchange. 6595 * @param tag IO tag value. 6596 * @param cq_id The id of the completion queue where the WQE response is sent. 6597 * @param rnode Destination of ELS request (that is, the remote node). 6598 * 6599 * @return Returns 0 on success, or a non-zero value on failure. 6600 */ 6601 int32_t 6602 sli_els_request64_wqe(sli4_t *sli4, void *buf, size_t size, ocs_dma_t *sgl, uint8_t req_type, 6603 uint32_t req_len, uint32_t max_rsp_len, uint8_t timeout, 6604 uint16_t xri, uint16_t tag, uint16_t cq_id, ocs_remote_node_t *rnode) 6605 { 6606 sli4_els_request64_wqe_t *els = buf; 6607 sli4_sge_t *sge = sgl->virt; 6608 uint8_t is_fabric = FALSE; 6609 6610 ocs_memset(buf, 0, size); 6611 6612 if (sli4->config.sgl_pre_registered) { 6613 els->xbl = FALSE; 6614 6615 els->dbde = TRUE; 6616 els->els_request_payload.bde_type = SLI4_BDE_TYPE_BDE_64; 6617 6618 els->els_request_payload.buffer_length = req_len; 6619 els->els_request_payload.u.data.buffer_address_low = sge[0].buffer_address_low; 6620 els->els_request_payload.u.data.buffer_address_high = sge[0].buffer_address_high; 6621 } else { 6622 els->xbl = TRUE; 6623 6624 els->els_request_payload.bde_type = SLI4_BDE_TYPE_BLP; 6625 6626 els->els_request_payload.buffer_length = 2 * sizeof(sli4_sge_t); 6627 els->els_request_payload.u.blp.sgl_segment_address_low = ocs_addr32_lo(sgl->phys); 6628 els->els_request_payload.u.blp.sgl_segment_address_high = ocs_addr32_hi(sgl->phys); 6629 } 6630 6631 els->els_request_payload_length = req_len; 6632 els->max_response_payload_length = max_rsp_len; 6633 6634 els->xri_tag = xri; 6635 els->timer = timeout; 6636 els->class = SLI4_ELS_REQUEST64_CLASS_3; 6637 6638 els->command = SLI4_WQE_ELS_REQUEST64; 6639 6640 els->request_tag = tag; 6641 6642 if (rnode->node_group) { 6643 els->hlm = TRUE; 6644 els->remote_id = rnode->fc_id & 0x00ffffff; 6645 } 6646 6647 els->iod = SLI4_ELS_REQUEST64_DIR_READ; 6648 6649 els->qosd = TRUE; 6650 6651 /* figure out the ELS_ID value from the request buffer */ 6652 6653 switch (req_type) { 6654 case FC_ELS_CMD_LOGO: 6655 els->els_id = SLI4_ELS_REQUEST64_LOGO; 6656 if (rnode->attached) { 6657 els->ct = SLI4_ELS_REQUEST64_CONTEXT_RPI; 6658 els->context_tag = rnode->indicator; 6659 } else { 6660 els->ct = SLI4_ELS_REQUEST64_CONTEXT_VPI; 6661 els->context_tag = rnode->sport->indicator; 6662 } 6663 if (FC_ADDR_FABRIC == rnode->fc_id) { 6664 is_fabric = TRUE; 6665 } 6666 break; 6667 case FC_ELS_CMD_FDISC: 6668 if (FC_ADDR_FABRIC == rnode->fc_id) { 6669 is_fabric = TRUE; 6670 } 6671 if (0 == rnode->sport->fc_id) { 6672 els->els_id = SLI4_ELS_REQUEST64_FDISC; 6673 is_fabric = TRUE; 6674 } else { 6675 els->els_id = SLI4_ELS_REQUEST64_OTHER; 6676 } 6677 els->ct = SLI4_ELS_REQUEST64_CONTEXT_VPI; 6678 els->context_tag = rnode->sport->indicator; 6679 els->sp = TRUE; 6680 break; 6681 case FC_ELS_CMD_FLOGI: 6682 els->els_id = SLI4_ELS_REQUEST64_FLOGIN; 6683 is_fabric = TRUE; 6684 if (SLI4_IF_TYPE_BE3_SKH_PF == sli4->if_type) { 6685 if (!rnode->sport->domain) { 6686 ocs_log_test(sli4->os, "invalid domain handle\n"); 6687 return -1; 6688 } 6689 /* 6690 * IF_TYPE 0 skips INIT_VFI/INIT_VPI and therefore must use the 6691 * FCFI here 6692 */ 6693 els->ct = SLI4_ELS_REQUEST64_CONTEXT_FCFI; 6694 els->context_tag = rnode->sport->domain->fcf_indicator; 6695 els->sp = TRUE; 6696 } else { 6697 els->ct = SLI4_ELS_REQUEST64_CONTEXT_VPI; 6698 els->context_tag = rnode->sport->indicator; 6699 6700 /* 6701 * Set SP here ... we haven't done a REG_VPI yet 6702 * TODO: need to maybe not set this when we have 6703 * completed VFI/VPI registrations ... 6704 * 6705 * Use the FC_ID of the SPORT if it has been allocated, otherwise 6706 * use an S_ID of zero. 6707 */ 6708 els->sp = TRUE; 6709 if (rnode->sport->fc_id != UINT32_MAX) { 6710 els->sid = rnode->sport->fc_id; 6711 } 6712 } 6713 break; 6714 case FC_ELS_CMD_PLOGI: 6715 els->els_id = SLI4_ELS_REQUEST64_PLOGI; 6716 els->ct = SLI4_ELS_REQUEST64_CONTEXT_VPI; 6717 els->context_tag = rnode->sport->indicator; 6718 break; 6719 case FC_ELS_CMD_SCR: 6720 els->els_id = SLI4_ELS_REQUEST64_OTHER; 6721 els->ct = SLI4_ELS_REQUEST64_CONTEXT_VPI; 6722 els->context_tag = rnode->sport->indicator; 6723 break; 6724 default: 6725 els->els_id = SLI4_ELS_REQUEST64_OTHER; 6726 if (rnode->attached) { 6727 els->ct = SLI4_ELS_REQUEST64_CONTEXT_RPI; 6728 els->context_tag = rnode->indicator; 6729 } else { 6730 els->ct = SLI4_ELS_REQUEST64_CONTEXT_VPI; 6731 els->context_tag = rnode->sport->indicator; 6732 } 6733 break; 6734 } 6735 6736 if (is_fabric) { 6737 els->cmd_type = SLI4_ELS_REQUEST64_CMD_FABRIC; 6738 } else { 6739 els->cmd_type = SLI4_ELS_REQUEST64_CMD_NON_FABRIC; 6740 } 6741 6742 els->cq_id = cq_id; 6743 6744 if (SLI4_ELS_REQUEST64_CONTEXT_RPI != els->ct) { 6745 els->remote_id = rnode->fc_id; 6746 } 6747 if (SLI4_ELS_REQUEST64_CONTEXT_VPI == els->ct) { 6748 els->temporary_rpi = rnode->indicator; 6749 } 6750 6751 return 0; 6752 } 6753 6754 /** 6755 * @ingroup sli_fc 6756 * @brief Write an FCP_ICMND64_WQE work queue entry. 6757 * 6758 * @param sli4 SLI context. 6759 * @param buf Destination buffer for the WQE. 6760 * @param size Buffer size, in bytes. 6761 * @param sgl DMA memory for the scatter gather list. 6762 * @param xri XRI for this exchange. 6763 * @param tag IO tag value. 6764 * @param cq_id The id of the completion queue where the WQE response is sent. 6765 * @param rpi remote node indicator (RPI) 6766 * @param rnode Destination request (that is, the remote node). 6767 * @param timeout Time, in seconds, before an IO times out. Zero means no timeout. 6768 * 6769 * @return Returns 0 on success, or a non-zero value on failure. 6770 */ 6771 int32_t 6772 sli_fcp_icmnd64_wqe(sli4_t *sli4, void *buf, size_t size, ocs_dma_t *sgl, 6773 uint16_t xri, uint16_t tag, uint16_t cq_id, 6774 uint32_t rpi, ocs_remote_node_t *rnode, uint8_t timeout) 6775 { 6776 sli4_fcp_icmnd64_wqe_t *icmnd = buf; 6777 sli4_sge_t *sge = NULL; 6778 6779 ocs_memset(buf, 0, size); 6780 6781 if (!sgl || !sgl->virt) { 6782 ocs_log_err(sli4->os, "bad parameter sgl=%p virt=%p\n", 6783 sgl, sgl ? sgl->virt : NULL); 6784 return -1; 6785 } 6786 sge = sgl->virt; 6787 6788 if (sli4->config.sgl_pre_registered) { 6789 icmnd->xbl = FALSE; 6790 6791 icmnd->dbde = TRUE; 6792 icmnd->bde.bde_type = SLI4_BDE_TYPE_BDE_64; 6793 6794 icmnd->bde.buffer_length = sge[0].buffer_length; 6795 icmnd->bde.u.data.buffer_address_low = sge[0].buffer_address_low; 6796 icmnd->bde.u.data.buffer_address_high = sge[0].buffer_address_high; 6797 } else { 6798 icmnd->xbl = TRUE; 6799 6800 icmnd->bde.bde_type = SLI4_BDE_TYPE_BLP; 6801 6802 icmnd->bde.buffer_length = sgl->size; 6803 icmnd->bde.u.blp.sgl_segment_address_low = ocs_addr32_lo(sgl->phys); 6804 icmnd->bde.u.blp.sgl_segment_address_high = ocs_addr32_hi(sgl->phys); 6805 } 6806 6807 icmnd->payload_offset_length = sge[0].buffer_length + sge[1].buffer_length; 6808 icmnd->xri_tag = xri; 6809 icmnd->context_tag = rpi; 6810 icmnd->timer = timeout; 6811 6812 icmnd->pu = 2; /* WQE word 4 contains read transfer length */ 6813 icmnd->class = SLI4_ELS_REQUEST64_CLASS_3; 6814 icmnd->command = SLI4_WQE_FCP_ICMND64; 6815 icmnd->ct = SLI4_ELS_REQUEST64_CONTEXT_RPI; 6816 6817 icmnd->abort_tag = xri; 6818 6819 icmnd->request_tag = tag; 6820 icmnd->len_loc = 3; 6821 if (rnode->node_group) { 6822 icmnd->hlm = TRUE; 6823 icmnd->remote_n_port_id = rnode->fc_id & 0x00ffffff; 6824 } 6825 if (((ocs_node_t *)rnode->node)->fcp2device) { 6826 icmnd->erp = TRUE; 6827 } 6828 icmnd->cmd_type = SLI4_CMD_FCP_ICMND64_WQE; 6829 icmnd->cq_id = cq_id; 6830 6831 return 0; 6832 } 6833 6834 /** 6835 * @ingroup sli_fc 6836 * @brief Write an FCP_IREAD64_WQE work queue entry. 6837 * 6838 * @param sli4 SLI context. 6839 * @param buf Destination buffer for the WQE. 6840 * @param size Buffer size, in bytes. 6841 * @param sgl DMA memory for the scatter gather list. 6842 * @param first_data_sge Index of first data sge (used if perf hints are enabled) 6843 * @param xfer_len Data transfer length. 6844 * @param xri XRI for this exchange. 6845 * @param tag IO tag value. 6846 * @param cq_id The id of the completion queue where the WQE response is sent. 6847 * @param rpi remote node indicator (RPI) 6848 * @param rnode Destination request (i.e. remote node). 6849 * @param dif T10 DIF operation, or 0 to disable. 6850 * @param bs T10 DIF block size, or 0 if DIF is disabled. 6851 * @param timeout Time, in seconds, before an IO times out. Zero means no timeout. 6852 * 6853 * @return Returns 0 on success, or a non-zero value on failure. 6854 */ 6855 int32_t 6856 sli_fcp_iread64_wqe(sli4_t *sli4, void *buf, size_t size, ocs_dma_t *sgl, uint32_t first_data_sge, 6857 uint32_t xfer_len, uint16_t xri, uint16_t tag, uint16_t cq_id, 6858 uint32_t rpi, ocs_remote_node_t *rnode, 6859 uint8_t dif, uint8_t bs, uint8_t timeout) 6860 { 6861 sli4_fcp_iread64_wqe_t *iread = buf; 6862 sli4_sge_t *sge = NULL; 6863 6864 ocs_memset(buf, 0, size); 6865 6866 if (!sgl || !sgl->virt) { 6867 ocs_log_err(sli4->os, "bad parameter sgl=%p virt=%p\n", 6868 sgl, sgl ? sgl->virt : NULL); 6869 return -1; 6870 } 6871 sge = sgl->virt; 6872 6873 if (sli4->config.sgl_pre_registered) { 6874 iread->xbl = FALSE; 6875 6876 iread->dbde = TRUE; 6877 iread->bde.bde_type = SLI4_BDE_TYPE_BDE_64; 6878 6879 iread->bde.buffer_length = sge[0].buffer_length; 6880 iread->bde.u.data.buffer_address_low = sge[0].buffer_address_low; 6881 iread->bde.u.data.buffer_address_high = sge[0].buffer_address_high; 6882 } else { 6883 iread->xbl = TRUE; 6884 6885 iread->bde.bde_type = SLI4_BDE_TYPE_BLP; 6886 6887 iread->bde.buffer_length = sgl->size; 6888 iread->bde.u.blp.sgl_segment_address_low = ocs_addr32_lo(sgl->phys); 6889 iread->bde.u.blp.sgl_segment_address_high = ocs_addr32_hi(sgl->phys); 6890 6891 /* fill out fcp_cmnd buffer len and change resp buffer to be of type 6892 * "skip" (note: response will still be written to sge[1] if necessary) */ 6893 iread->fcp_cmd_buffer_length = sge[0].buffer_length; 6894 sge[1].sge_type = SLI4_SGE_TYPE_SKIP; 6895 } 6896 6897 iread->payload_offset_length = sge[0].buffer_length + sge[1].buffer_length; 6898 iread->total_transfer_length = xfer_len; 6899 6900 iread->xri_tag = xri; 6901 iread->context_tag = rpi; 6902 6903 iread->timer = timeout; 6904 6905 iread->pu = 2; /* WQE word 4 contains read transfer length */ 6906 iread->class = SLI4_ELS_REQUEST64_CLASS_3; 6907 iread->command = SLI4_WQE_FCP_IREAD64; 6908 iread->ct = SLI4_ELS_REQUEST64_CONTEXT_RPI; 6909 iread->dif = dif; 6910 iread->bs = bs; 6911 6912 iread->abort_tag = xri; 6913 6914 iread->request_tag = tag; 6915 iread->len_loc = 3; 6916 if (rnode->node_group) { 6917 iread->hlm = TRUE; 6918 iread->remote_n_port_id = rnode->fc_id & 0x00ffffff; 6919 } 6920 if (((ocs_node_t *)rnode->node)->fcp2device) { 6921 iread->erp = TRUE; 6922 } 6923 iread->iod = 1; 6924 iread->cmd_type = SLI4_CMD_FCP_IREAD64_WQE; 6925 iread->cq_id = cq_id; 6926 6927 if (sli4->config.perf_hint) { 6928 iread->first_data_bde.bde_type = SLI4_BDE_TYPE_BDE_64; 6929 iread->first_data_bde.buffer_length = sge[first_data_sge].buffer_length; 6930 iread->first_data_bde.u.data.buffer_address_low = sge[first_data_sge].buffer_address_low; 6931 iread->first_data_bde.u.data.buffer_address_high = sge[first_data_sge].buffer_address_high; 6932 } 6933 6934 return 0; 6935 } 6936 6937 /** 6938 * @ingroup sli_fc 6939 * @brief Write an FCP_IWRITE64_WQE work queue entry. 6940 * 6941 * @param sli4 SLI context. 6942 * @param buf Destination buffer for the WQE. 6943 * @param size Buffer size, in bytes. 6944 * @param sgl DMA memory for the scatter gather list. 6945 * @param first_data_sge Index of first data sge (used if perf hints are enabled) 6946 * @param xfer_len Data transfer length. 6947 * @param first_burst The number of first burst bytes 6948 * @param xri XRI for this exchange. 6949 * @param tag IO tag value. 6950 * @param cq_id The id of the completion queue where the WQE response is sent. 6951 * @param rpi remote node indicator (RPI) 6952 * @param rnode Destination request (i.e. remote node) 6953 * @param dif T10 DIF operation, or 0 to disable 6954 * @param bs T10 DIF block size, or 0 if DIF is disabled 6955 * @param timeout Time, in seconds, before an IO times out. Zero means no timeout. 6956 * 6957 * @return Returns 0 on success, or a non-zero value on failure. 6958 */ 6959 int32_t 6960 sli_fcp_iwrite64_wqe(sli4_t *sli4, void *buf, size_t size, ocs_dma_t *sgl, uint32_t first_data_sge, 6961 uint32_t xfer_len, uint32_t first_burst, uint16_t xri, uint16_t tag, uint16_t cq_id, 6962 uint32_t rpi, ocs_remote_node_t *rnode, 6963 uint8_t dif, uint8_t bs, uint8_t timeout) 6964 { 6965 sli4_fcp_iwrite64_wqe_t *iwrite = buf; 6966 sli4_sge_t *sge = NULL; 6967 6968 ocs_memset(buf, 0, size); 6969 6970 if (!sgl || !sgl->virt) { 6971 ocs_log_err(sli4->os, "bad parameter sgl=%p virt=%p\n", 6972 sgl, sgl ? sgl->virt : NULL); 6973 return -1; 6974 } 6975 sge = sgl->virt; 6976 6977 if (sli4->config.sgl_pre_registered) { 6978 iwrite->xbl = FALSE; 6979 6980 iwrite->dbde = TRUE; 6981 iwrite->bde.bde_type = SLI4_BDE_TYPE_BDE_64; 6982 6983 iwrite->bde.buffer_length = sge[0].buffer_length; 6984 iwrite->bde.u.data.buffer_address_low = sge[0].buffer_address_low; 6985 iwrite->bde.u.data.buffer_address_high = sge[0].buffer_address_high; 6986 } else { 6987 iwrite->xbl = TRUE; 6988 6989 iwrite->bde.bde_type = SLI4_BDE_TYPE_BLP; 6990 6991 iwrite->bde.buffer_length = sgl->size; 6992 iwrite->bde.u.blp.sgl_segment_address_low = ocs_addr32_lo(sgl->phys); 6993 iwrite->bde.u.blp.sgl_segment_address_high = ocs_addr32_hi(sgl->phys); 6994 6995 /* fill out fcp_cmnd buffer len and change resp buffer to be of type 6996 * "skip" (note: response will still be written to sge[1] if necessary) */ 6997 iwrite->fcp_cmd_buffer_length = sge[0].buffer_length; 6998 sge[1].sge_type = SLI4_SGE_TYPE_SKIP; 6999 } 7000 7001 iwrite->payload_offset_length = sge[0].buffer_length + sge[1].buffer_length; 7002 iwrite->total_transfer_length = xfer_len; 7003 iwrite->initial_transfer_length = MIN(xfer_len, first_burst); 7004 7005 iwrite->xri_tag = xri; 7006 iwrite->context_tag = rpi; 7007 7008 iwrite->timer = timeout; 7009 7010 iwrite->pu = 2; /* WQE word 4 contains read transfer length */ 7011 iwrite->class = SLI4_ELS_REQUEST64_CLASS_3; 7012 iwrite->command = SLI4_WQE_FCP_IWRITE64; 7013 iwrite->ct = SLI4_ELS_REQUEST64_CONTEXT_RPI; 7014 iwrite->dif = dif; 7015 iwrite->bs = bs; 7016 7017 iwrite->abort_tag = xri; 7018 7019 iwrite->request_tag = tag; 7020 iwrite->len_loc = 3; 7021 if (rnode->node_group) { 7022 iwrite->hlm = TRUE; 7023 iwrite->remote_n_port_id = rnode->fc_id & 0x00ffffff; 7024 } 7025 if (((ocs_node_t *)rnode->node)->fcp2device) { 7026 iwrite->erp = TRUE; 7027 } 7028 iwrite->cmd_type = SLI4_CMD_FCP_IWRITE64_WQE; 7029 iwrite->cq_id = cq_id; 7030 7031 if (sli4->config.perf_hint) { 7032 iwrite->first_data_bde.bde_type = SLI4_BDE_TYPE_BDE_64; 7033 iwrite->first_data_bde.buffer_length = sge[first_data_sge].buffer_length; 7034 iwrite->first_data_bde.u.data.buffer_address_low = sge[first_data_sge].buffer_address_low; 7035 iwrite->first_data_bde.u.data.buffer_address_high = sge[first_data_sge].buffer_address_high; 7036 } 7037 7038 return 0; 7039 } 7040 7041 /** 7042 * @ingroup sli_fc 7043 * @brief Write an FCP_TRECEIVE64_WQE work queue entry. 7044 * 7045 * @param sli4 SLI context. 7046 * @param buf Destination buffer for the WQE. 7047 * @param size Buffer size, in bytes. 7048 * @param sgl DMA memory for the Scatter-Gather List. 7049 * @param first_data_sge Index of first data sge (used if perf hints are enabled) 7050 * @param relative_off Relative offset of the IO (if any). 7051 * @param xfer_len Data transfer length. 7052 * @param xri XRI for this exchange. 7053 * @param tag IO tag value. 7054 * @param xid OX_ID for the exchange. 7055 * @param cq_id The id of the completion queue where the WQE response is sent. 7056 * @param rpi remote node indicator (RPI) 7057 * @param rnode Destination request (i.e. remote node). 7058 * @param flags Optional attributes, including: 7059 * - ACTIVE - IO is already active. 7060 * - AUTO RSP - Automatically generate a good FCP_RSP. 7061 * @param dif T10 DIF operation, or 0 to disable. 7062 * @param bs T10 DIF block size, or 0 if DIF is disabled. 7063 * @param csctl value of csctl field. 7064 * @param app_id value for VM application header. 7065 * 7066 * @return Returns 0 on success, or a non-zero value on failure. 7067 */ 7068 int32_t 7069 sli_fcp_treceive64_wqe(sli4_t *sli4, void *buf, size_t size, ocs_dma_t *sgl, uint32_t first_data_sge, 7070 uint32_t relative_off, uint32_t xfer_len, uint16_t xri, uint16_t tag, uint16_t cq_id, 7071 uint16_t xid, uint32_t rpi, ocs_remote_node_t *rnode, uint32_t flags, uint8_t dif, uint8_t bs, 7072 uint8_t csctl, uint32_t app_id) 7073 { 7074 sli4_fcp_treceive64_wqe_t *trecv = buf; 7075 sli4_fcp_128byte_wqe_t *trecv_128 = buf; 7076 sli4_sge_t *sge = NULL; 7077 7078 ocs_memset(buf, 0, size); 7079 7080 if (!sgl || !sgl->virt) { 7081 ocs_log_err(sli4->os, "bad parameter sgl=%p virt=%p\n", 7082 sgl, sgl ? sgl->virt : NULL); 7083 return -1; 7084 } 7085 sge = sgl->virt; 7086 7087 if (sli4->config.sgl_pre_registered) { 7088 trecv->xbl = FALSE; 7089 7090 trecv->dbde = TRUE; 7091 trecv->bde.bde_type = SLI4_BDE_TYPE_BDE_64; 7092 7093 trecv->bde.buffer_length = sge[0].buffer_length; 7094 trecv->bde.u.data.buffer_address_low = sge[0].buffer_address_low; 7095 trecv->bde.u.data.buffer_address_high = sge[0].buffer_address_high; 7096 7097 trecv->payload_offset_length = sge[0].buffer_length; 7098 } else { 7099 trecv->xbl = TRUE; 7100 7101 /* if data is a single physical address, use a BDE */ 7102 if (!dif && (xfer_len <= sge[2].buffer_length)) { 7103 trecv->dbde = TRUE; 7104 trecv->bde.bde_type = SLI4_BDE_TYPE_BDE_64; 7105 7106 trecv->bde.buffer_length = sge[2].buffer_length; 7107 trecv->bde.u.data.buffer_address_low = sge[2].buffer_address_low; 7108 trecv->bde.u.data.buffer_address_high = sge[2].buffer_address_high; 7109 } else { 7110 trecv->bde.bde_type = SLI4_BDE_TYPE_BLP; 7111 trecv->bde.buffer_length = sgl->size; 7112 trecv->bde.u.blp.sgl_segment_address_low = ocs_addr32_lo(sgl->phys); 7113 trecv->bde.u.blp.sgl_segment_address_high = ocs_addr32_hi(sgl->phys); 7114 } 7115 } 7116 7117 trecv->relative_offset = relative_off; 7118 7119 if (flags & SLI4_IO_CONTINUATION) { 7120 trecv->xc = TRUE; 7121 } 7122 trecv->xri_tag = xri; 7123 7124 trecv->context_tag = rpi; 7125 7126 trecv->pu = TRUE; /* WQE uses relative offset */ 7127 7128 if (flags & SLI4_IO_AUTO_GOOD_RESPONSE) { 7129 trecv->ar = TRUE; 7130 } 7131 7132 trecv->command = SLI4_WQE_FCP_TRECEIVE64; 7133 trecv->class = SLI4_ELS_REQUEST64_CLASS_3; 7134 trecv->ct = SLI4_ELS_REQUEST64_CONTEXT_RPI; 7135 trecv->dif = dif; 7136 trecv->bs = bs; 7137 7138 trecv->remote_xid = xid; 7139 7140 trecv->request_tag = tag; 7141 7142 trecv->iod = 1; 7143 7144 trecv->len_loc = 0x2; 7145 7146 if (rnode->node_group) { 7147 trecv->hlm = TRUE; 7148 trecv->dword5.dword = rnode->fc_id & 0x00ffffff; 7149 } 7150 7151 trecv->cmd_type = SLI4_CMD_FCP_TRECEIVE64_WQE; 7152 7153 trecv->cq_id = cq_id; 7154 7155 trecv->fcp_data_receive_length = xfer_len; 7156 7157 if (sli4->config.perf_hint) { 7158 trecv->first_data_bde.bde_type = SLI4_BDE_TYPE_BDE_64; 7159 trecv->first_data_bde.buffer_length = sge[first_data_sge].buffer_length; 7160 trecv->first_data_bde.u.data.buffer_address_low = sge[first_data_sge].buffer_address_low; 7161 trecv->first_data_bde.u.data.buffer_address_high = sge[first_data_sge].buffer_address_high; 7162 } 7163 7164 /* The upper 7 bits of csctl is the priority */ 7165 if (csctl & SLI4_MASK_CCP) { 7166 trecv->ccpe = 1; 7167 trecv->ccp = (csctl & SLI4_MASK_CCP); 7168 } 7169 7170 if (app_id && (sli4->config.wqe_size == SLI4_WQE_EXT_BYTES) && !trecv->eat) { 7171 trecv->app_id_valid = 1; 7172 trecv->wqes = 1; 7173 trecv_128->dw[31] = app_id; 7174 } 7175 return 0; 7176 } 7177 7178 /** 7179 * @ingroup sli_fc 7180 * @brief Write an FCP_CONT_TRECEIVE64_WQE work queue entry. 7181 * 7182 * @param sli4 SLI context. 7183 * @param buf Destination buffer for the WQE. 7184 * @param size Buffer size, in bytes. 7185 * @param sgl DMA memory for the Scatter-Gather List. 7186 * @param first_data_sge Index of first data sge (used if perf hints are enabled) 7187 * @param relative_off Relative offset of the IO (if any). 7188 * @param xfer_len Data transfer length. 7189 * @param xri XRI for this exchange. 7190 * @param sec_xri Secondary XRI for this exchange. (BZ 161832 workaround) 7191 * @param tag IO tag value. 7192 * @param xid OX_ID for the exchange. 7193 * @param cq_id The id of the completion queue where the WQE response is sent. 7194 * @param rpi remote node indicator (RPI) 7195 * @param rnode Destination request (i.e. remote node). 7196 * @param flags Optional attributes, including: 7197 * - ACTIVE - IO is already active. 7198 * - AUTO RSP - Automatically generate a good FCP_RSP. 7199 * @param dif T10 DIF operation, or 0 to disable. 7200 * @param bs T10 DIF block size, or 0 if DIF is disabled. 7201 * @param csctl value of csctl field. 7202 * @param app_id value for VM application header. 7203 * 7204 * @return Returns 0 on success, or a non-zero value on failure. 7205 */ 7206 int32_t 7207 sli_fcp_cont_treceive64_wqe(sli4_t *sli4, void *buf, size_t size, ocs_dma_t *sgl, uint32_t first_data_sge, 7208 uint32_t relative_off, uint32_t xfer_len, uint16_t xri, uint16_t sec_xri, uint16_t tag, 7209 uint16_t cq_id, uint16_t xid, uint32_t rpi, ocs_remote_node_t *rnode, uint32_t flags, 7210 uint8_t dif, uint8_t bs, uint8_t csctl, uint32_t app_id) 7211 { 7212 int32_t rc; 7213 7214 rc = sli_fcp_treceive64_wqe(sli4, buf, size, sgl, first_data_sge, relative_off, xfer_len, xri, tag, 7215 cq_id, xid, rpi, rnode, flags, dif, bs, csctl, app_id); 7216 if (rc == 0) { 7217 sli4_fcp_treceive64_wqe_t *trecv = buf; 7218 7219 trecv->command = SLI4_WQE_FCP_CONT_TRECEIVE64; 7220 trecv->dword5.sec_xri_tag = sec_xri; 7221 } 7222 return rc; 7223 } 7224 7225 /** 7226 * @ingroup sli_fc 7227 * @brief Write an FCP_TRSP64_WQE work queue entry. 7228 * 7229 * @param sli4 SLI context. 7230 * @param buf Destination buffer for the WQE. 7231 * @param size Buffer size, in bytes. 7232 * @param sgl DMA memory for the Scatter-Gather List. 7233 * @param rsp_len Response data length. 7234 * @param xri XRI for this exchange. 7235 * @param tag IO tag value. 7236 * @param cq_id The id of the completion queue where the WQE response is sent. 7237 * @param xid OX_ID for the exchange. 7238 * @param rpi remote node indicator (RPI) 7239 * @param rnode Destination request (i.e. remote node). 7240 * @param flags Optional attributes, including: 7241 * - ACTIVE - IO is already active 7242 * - AUTO RSP - Automatically generate a good FCP_RSP. 7243 * @param csctl value of csctl field. 7244 * @param port_owned 0/1 to indicate if the XRI is port owned (used to set XBL=0) 7245 * @param app_id value for VM application header. 7246 * 7247 * @return Returns 0 on success, or a non-zero value on failure. 7248 */ 7249 int32_t 7250 sli_fcp_trsp64_wqe(sli4_t *sli4, void *buf, size_t size, ocs_dma_t *sgl, uint32_t rsp_len, 7251 uint16_t xri, uint16_t tag, uint16_t cq_id, uint16_t xid, uint32_t rpi, ocs_remote_node_t *rnode, 7252 uint32_t flags, uint8_t csctl, uint8_t port_owned, uint32_t app_id) 7253 { 7254 sli4_fcp_trsp64_wqe_t *trsp = buf; 7255 sli4_fcp_128byte_wqe_t *trsp_128 = buf; 7256 7257 ocs_memset(buf, 0, size); 7258 7259 if (flags & SLI4_IO_AUTO_GOOD_RESPONSE) { 7260 trsp->ag = TRUE; 7261 /* 7262 * The SLI-4 documentation states that the BDE is ignored when 7263 * using auto-good response, but, at least for IF_TYPE 0 devices, 7264 * this does not appear to be true. 7265 */ 7266 if (SLI4_IF_TYPE_BE3_SKH_PF == sli4->if_type) { 7267 trsp->bde.buffer_length = 12; /* byte size of RSP */ 7268 } 7269 } else { 7270 sli4_sge_t *sge = sgl->virt; 7271 7272 if (sli4->config.sgl_pre_registered || port_owned) { 7273 trsp->dbde = TRUE; 7274 } else { 7275 trsp->xbl = TRUE; 7276 } 7277 7278 trsp->bde.bde_type = SLI4_BDE_TYPE_BDE_64; 7279 trsp->bde.buffer_length = sge[0].buffer_length; 7280 trsp->bde.u.data.buffer_address_low = sge[0].buffer_address_low; 7281 trsp->bde.u.data.buffer_address_high = sge[0].buffer_address_high; 7282 7283 trsp->fcp_response_length = rsp_len; 7284 } 7285 7286 if (flags & SLI4_IO_CONTINUATION) { 7287 trsp->xc = TRUE; 7288 } 7289 7290 if (rnode->node_group) { 7291 trsp->hlm = TRUE; 7292 trsp->dword5 = rnode->fc_id & 0x00ffffff; 7293 } 7294 7295 trsp->xri_tag = xri; 7296 trsp->rpi = rpi; 7297 7298 trsp->command = SLI4_WQE_FCP_TRSP64; 7299 trsp->class = SLI4_ELS_REQUEST64_CLASS_3; 7300 7301 trsp->remote_xid = xid; 7302 trsp->request_tag = tag; 7303 trsp->dnrx = ((flags & SLI4_IO_DNRX) == 0 ? 0 : 1); 7304 trsp->len_loc = 0x1; 7305 trsp->cq_id = cq_id; 7306 trsp->cmd_type = SLI4_CMD_FCP_TRSP64_WQE; 7307 7308 /* The upper 7 bits of csctl is the priority */ 7309 if (csctl & SLI4_MASK_CCP) { 7310 trsp->ccpe = 1; 7311 trsp->ccp = (csctl & SLI4_MASK_CCP); 7312 } 7313 7314 if (app_id && (sli4->config.wqe_size == SLI4_WQE_EXT_BYTES) && !trsp->eat) { 7315 trsp->app_id_valid = 1; 7316 trsp->wqes = 1; 7317 trsp_128->dw[31] = app_id; 7318 } 7319 return 0; 7320 } 7321 7322 /** 7323 * @ingroup sli_fc 7324 * @brief Write an FCP_TSEND64_WQE work queue entry. 7325 * 7326 * @param sli4 SLI context. 7327 * @param buf Destination buffer for the WQE. 7328 * @param size Buffer size, in bytes. 7329 * @param sgl DMA memory for the scatter gather list. 7330 * @param first_data_sge Index of first data sge (used if perf hints are enabled) 7331 * @param relative_off Relative offset of the IO (if any). 7332 * @param xfer_len Data transfer length. 7333 * @param xri XRI for this exchange. 7334 * @param tag IO tag value. 7335 * @param cq_id The id of the completion queue where the WQE response is sent. 7336 * @param xid OX_ID for the exchange. 7337 * @param rpi remote node indicator (RPI) 7338 * @param rnode Destination request (i.e. remote node). 7339 * @param flags Optional attributes, including: 7340 * - ACTIVE - IO is already active. 7341 * - AUTO RSP - Automatically generate a good FCP_RSP. 7342 * @param dif T10 DIF operation, or 0 to disable. 7343 * @param bs T10 DIF block size, or 0 if DIF is disabled. 7344 * @param csctl value of csctl field. 7345 * @param app_id value for VM application header. 7346 * 7347 * @return Returns 0 on success, or a non-zero value on failure. 7348 */ 7349 int32_t 7350 sli_fcp_tsend64_wqe(sli4_t *sli4, void *buf, size_t size, ocs_dma_t *sgl, uint32_t first_data_sge, 7351 uint32_t relative_off, uint32_t xfer_len, 7352 uint16_t xri, uint16_t tag, uint16_t cq_id, uint16_t xid, uint32_t rpi, ocs_remote_node_t *rnode, 7353 uint32_t flags, uint8_t dif, uint8_t bs, uint8_t csctl, uint32_t app_id) 7354 { 7355 sli4_fcp_tsend64_wqe_t *tsend = buf; 7356 sli4_fcp_128byte_wqe_t *tsend_128 = buf; 7357 sli4_sge_t *sge = NULL; 7358 7359 ocs_memset(buf, 0, size); 7360 7361 if (!sgl || !sgl->virt) { 7362 ocs_log_err(sli4->os, "bad parameter sgl=%p virt=%p\n", 7363 sgl, sgl ? sgl->virt : NULL); 7364 return -1; 7365 } 7366 sge = sgl->virt; 7367 7368 if (sli4->config.sgl_pre_registered) { 7369 tsend->xbl = FALSE; 7370 7371 tsend->dbde = TRUE; 7372 tsend->bde.bde_type = SLI4_BDE_TYPE_BDE_64; 7373 7374 /* TSEND64_WQE specifies first two SGE are skipped 7375 * (i.e. 3rd is valid) */ 7376 tsend->bde.buffer_length = sge[2].buffer_length; 7377 tsend->bde.u.data.buffer_address_low = sge[2].buffer_address_low; 7378 tsend->bde.u.data.buffer_address_high = sge[2].buffer_address_high; 7379 } else { 7380 tsend->xbl = TRUE; 7381 7382 /* if data is a single physical address, use a BDE */ 7383 if (!dif && (xfer_len <= sge[2].buffer_length)) { 7384 tsend->dbde = TRUE; 7385 tsend->bde.bde_type = SLI4_BDE_TYPE_BDE_64; 7386 /* TSEND64_WQE specifies first two SGE are skipped 7387 * (i.e. 3rd is valid) */ 7388 tsend->bde.buffer_length = sge[2].buffer_length; 7389 tsend->bde.u.data.buffer_address_low = sge[2].buffer_address_low; 7390 tsend->bde.u.data.buffer_address_high = sge[2].buffer_address_high; 7391 } else { 7392 tsend->bde.bde_type = SLI4_BDE_TYPE_BLP; 7393 tsend->bde.buffer_length = sgl->size; 7394 tsend->bde.u.blp.sgl_segment_address_low = ocs_addr32_lo(sgl->phys); 7395 tsend->bde.u.blp.sgl_segment_address_high = ocs_addr32_hi(sgl->phys); 7396 } 7397 } 7398 7399 tsend->relative_offset = relative_off; 7400 7401 if (flags & SLI4_IO_CONTINUATION) { 7402 tsend->xc = TRUE; 7403 } 7404 tsend->xri_tag = xri; 7405 7406 tsend->rpi = rpi; 7407 7408 tsend->pu = TRUE; /* WQE uses relative offset */ 7409 7410 if (flags & SLI4_IO_AUTO_GOOD_RESPONSE) { 7411 tsend->ar = TRUE; 7412 } 7413 7414 tsend->command = SLI4_WQE_FCP_TSEND64; 7415 tsend->class = SLI4_ELS_REQUEST64_CLASS_3; 7416 tsend->ct = SLI4_ELS_REQUEST64_CONTEXT_RPI; 7417 tsend->dif = dif; 7418 tsend->bs = bs; 7419 7420 tsend->remote_xid = xid; 7421 7422 tsend->request_tag = tag; 7423 7424 tsend->len_loc = 0x2; 7425 7426 if (rnode->node_group) { 7427 tsend->hlm = TRUE; 7428 tsend->dword5 = rnode->fc_id & 0x00ffffff; 7429 } 7430 7431 tsend->cq_id = cq_id; 7432 7433 tsend->cmd_type = SLI4_CMD_FCP_TSEND64_WQE; 7434 7435 tsend->fcp_data_transmit_length = xfer_len; 7436 7437 if (sli4->config.perf_hint) { 7438 tsend->first_data_bde.bde_type = SLI4_BDE_TYPE_BDE_64; 7439 tsend->first_data_bde.buffer_length = sge[first_data_sge].buffer_length; 7440 tsend->first_data_bde.u.data.buffer_address_low = sge[first_data_sge].buffer_address_low; 7441 tsend->first_data_bde.u.data.buffer_address_high = sge[first_data_sge].buffer_address_high; 7442 } 7443 7444 /* The upper 7 bits of csctl is the priority */ 7445 if (csctl & SLI4_MASK_CCP) { 7446 tsend->ccpe = 1; 7447 tsend->ccp = (csctl & SLI4_MASK_CCP); 7448 } 7449 7450 if (app_id && (sli4->config.wqe_size == SLI4_WQE_EXT_BYTES) && !tsend->eat) { 7451 tsend->app_id_valid = 1; 7452 tsend->wqes = 1; 7453 tsend_128->dw[31] = app_id; 7454 } 7455 return 0; 7456 } 7457 7458 /** 7459 * @ingroup sli_fc 7460 * @brief Write a GEN_REQUEST64 work queue entry. 7461 * 7462 * @note This WQE is only used to send FC-CT commands. 7463 * 7464 * @param sli4 SLI context. 7465 * @param buf Destination buffer for the WQE. 7466 * @param size Buffer size, in bytes. 7467 * @param sgl DMA memory for the request. 7468 * @param req_len Length of request. 7469 * @param max_rsp_len Max length of response. 7470 * @param timeout Time, in seconds, before an IO times out. Zero means infinite. 7471 * @param xri XRI for this exchange. 7472 * @param tag IO tag value. 7473 * @param cq_id The id of the completion queue where the WQE response is sent. 7474 * @param rnode Destination of request (that is, the remote node). 7475 * @param r_ctl R_CTL value for sequence. 7476 * @param type TYPE value for sequence. 7477 * @param df_ctl DF_CTL value for sequence. 7478 * 7479 * @return Returns 0 on success, or a non-zero value on failure. 7480 */ 7481 int32_t 7482 sli_gen_request64_wqe(sli4_t *sli4, void *buf, size_t size, ocs_dma_t *sgl, 7483 uint32_t req_len, uint32_t max_rsp_len, uint8_t timeout, 7484 uint16_t xri, uint16_t tag, uint16_t cq_id, ocs_remote_node_t *rnode, 7485 uint8_t r_ctl, uint8_t type, uint8_t df_ctl) 7486 { 7487 sli4_gen_request64_wqe_t *gen = buf; 7488 sli4_sge_t *sge = NULL; 7489 7490 ocs_memset(buf, 0, size); 7491 7492 if (!sgl || !sgl->virt) { 7493 ocs_log_err(sli4->os, "bad parameter sgl=%p virt=%p\n", 7494 sgl, sgl ? sgl->virt : NULL); 7495 return -1; 7496 } 7497 sge = sgl->virt; 7498 7499 if (sli4->config.sgl_pre_registered) { 7500 gen->xbl = FALSE; 7501 7502 gen->dbde = TRUE; 7503 gen->bde.bde_type = SLI4_BDE_TYPE_BDE_64; 7504 7505 gen->bde.buffer_length = req_len; 7506 gen->bde.u.data.buffer_address_low = sge[0].buffer_address_low; 7507 gen->bde.u.data.buffer_address_high = sge[0].buffer_address_high; 7508 } else { 7509 gen->xbl = TRUE; 7510 7511 gen->bde.bde_type = SLI4_BDE_TYPE_BLP; 7512 7513 gen->bde.buffer_length = 2 * sizeof(sli4_sge_t); 7514 gen->bde.u.blp.sgl_segment_address_low = ocs_addr32_lo(sgl->phys); 7515 gen->bde.u.blp.sgl_segment_address_high = ocs_addr32_hi(sgl->phys); 7516 } 7517 7518 gen->request_payload_length = req_len; 7519 gen->max_response_payload_length = max_rsp_len; 7520 7521 gen->df_ctl = df_ctl; 7522 gen->type = type; 7523 gen->r_ctl = r_ctl; 7524 7525 gen->xri_tag = xri; 7526 7527 gen->ct = SLI4_ELS_REQUEST64_CONTEXT_RPI; 7528 gen->context_tag = rnode->indicator; 7529 7530 gen->class = SLI4_ELS_REQUEST64_CLASS_3; 7531 7532 gen->command = SLI4_WQE_GEN_REQUEST64; 7533 7534 gen->timer = timeout; 7535 7536 gen->request_tag = tag; 7537 7538 gen->iod = SLI4_ELS_REQUEST64_DIR_READ; 7539 7540 gen->qosd = TRUE; 7541 7542 if (rnode->node_group) { 7543 gen->hlm = TRUE; 7544 gen->remote_n_port_id = rnode->fc_id & 0x00ffffff; 7545 } 7546 7547 gen->cmd_type = SLI4_CMD_GEN_REQUEST64_WQE; 7548 7549 gen->cq_id = cq_id; 7550 7551 return 0; 7552 } 7553 7554 /** 7555 * @ingroup sli_fc 7556 * @brief Write a SEND_FRAME work queue entry 7557 * 7558 * @param sli4 SLI context. 7559 * @param buf Destination buffer for the WQE. 7560 * @param size Buffer size, in bytes. 7561 * @param sof Start of frame value 7562 * @param eof End of frame value 7563 * @param hdr Pointer to FC header data 7564 * @param payload DMA memory for the payload. 7565 * @param req_len Length of payload. 7566 * @param timeout Time, in seconds, before an IO times out. Zero means infinite. 7567 * @param xri XRI for this exchange. 7568 * @param req_tag IO tag value. 7569 * 7570 * @return Returns 0 on success, or a non-zero value on failure. 7571 */ 7572 int32_t 7573 sli_send_frame_wqe(sli4_t *sli4, void *buf, size_t size, uint8_t sof, uint8_t eof, uint32_t *hdr, 7574 ocs_dma_t *payload, uint32_t req_len, uint8_t timeout, 7575 uint16_t xri, uint16_t req_tag) 7576 { 7577 sli4_send_frame_wqe_t *sf = buf; 7578 7579 ocs_memset(buf, 0, size); 7580 7581 sf->dbde = TRUE; 7582 sf->bde.buffer_length = req_len; 7583 sf->bde.u.data.buffer_address_low = ocs_addr32_lo(payload->phys); 7584 sf->bde.u.data.buffer_address_high = ocs_addr32_hi(payload->phys); 7585 7586 /* Copy FC header */ 7587 sf->fc_header_0_1[0] = hdr[0]; 7588 sf->fc_header_0_1[1] = hdr[1]; 7589 sf->fc_header_2_5[0] = hdr[2]; 7590 sf->fc_header_2_5[1] = hdr[3]; 7591 sf->fc_header_2_5[2] = hdr[4]; 7592 sf->fc_header_2_5[3] = hdr[5]; 7593 7594 sf->frame_length = req_len; 7595 7596 sf->xri_tag = xri; 7597 sf->pu = 0; 7598 sf->context_tag = 0; 7599 7600 sf->ct = 0; 7601 sf->command = SLI4_WQE_SEND_FRAME; 7602 sf->class = SLI4_ELS_REQUEST64_CLASS_3; 7603 sf->timer = timeout; 7604 7605 sf->request_tag = req_tag; 7606 sf->eof = eof; 7607 sf->sof = sof; 7608 7609 sf->qosd = 0; 7610 sf->lenloc = 1; 7611 sf->xc = 0; 7612 7613 sf->xbl = 1; 7614 7615 sf->cmd_type = SLI4_CMD_SEND_FRAME_WQE; 7616 sf->cq_id = 0xffff; 7617 7618 return 0; 7619 } 7620 7621 /** 7622 * @ingroup sli_fc 7623 * @brief Write a XMIT_SEQUENCE64 work queue entry. 7624 * 7625 * This WQE is used to send FC-CT response frames. 7626 * 7627 * @note This API implements a restricted use for this WQE, a TODO: would 7628 * include passing in sequence initiative, and full SGL's 7629 * 7630 * @param sli4 SLI context. 7631 * @param buf Destination buffer for the WQE. 7632 * @param size Buffer size, in bytes. 7633 * @param payload DMA memory for the request. 7634 * @param payload_len Length of request. 7635 * @param timeout Time, in seconds, before an IO times out. Zero means infinite. 7636 * @param ox_id originator exchange ID 7637 * @param xri XRI for this exchange. 7638 * @param tag IO tag value. 7639 * @param rnode Destination of request (that is, the remote node). 7640 * @param r_ctl R_CTL value for sequence. 7641 * @param type TYPE value for sequence. 7642 * @param df_ctl DF_CTL value for sequence. 7643 * 7644 * @return Returns 0 on success, or a non-zero value on failure. 7645 */ 7646 int32_t 7647 sli_xmit_sequence64_wqe(sli4_t *sli4, void *buf, size_t size, ocs_dma_t *payload, 7648 uint32_t payload_len, uint8_t timeout, uint16_t ox_id, 7649 uint16_t xri, uint16_t tag, ocs_remote_node_t *rnode, 7650 uint8_t r_ctl, uint8_t type, uint8_t df_ctl) 7651 { 7652 sli4_xmit_sequence64_wqe_t *xmit = buf; 7653 7654 ocs_memset(buf, 0, size); 7655 7656 if ((payload == NULL) || (payload->virt == NULL)) { 7657 ocs_log_err(sli4->os, "bad parameter sgl=%p virt=%p\n", 7658 payload, payload ? payload->virt : NULL); 7659 return -1; 7660 } 7661 7662 if (sli4->config.sgl_pre_registered) { 7663 xmit->dbde = TRUE; 7664 } else { 7665 xmit->xbl = TRUE; 7666 } 7667 7668 xmit->bde.bde_type = SLI4_BDE_TYPE_BDE_64; 7669 xmit->bde.buffer_length = payload_len; 7670 xmit->bde.u.data.buffer_address_low = ocs_addr32_lo(payload->phys); 7671 xmit->bde.u.data.buffer_address_high = ocs_addr32_hi(payload->phys); 7672 xmit->sequence_payload_len = payload_len; 7673 7674 xmit->remote_n_port_id = rnode->fc_id & 0x00ffffff; 7675 7676 xmit->relative_offset = 0; 7677 7678 xmit->si = 0; /* sequence initiative - this matches what is seen from 7679 * FC switches in response to FCGS commands */ 7680 xmit->ft = 0; /* force transmit */ 7681 xmit->xo = 0; /* exchange responder */ 7682 xmit->ls = 1; /* last in seqence */ 7683 xmit->df_ctl = df_ctl; 7684 xmit->type = type; 7685 xmit->r_ctl = r_ctl; 7686 7687 xmit->xri_tag = xri; 7688 xmit->context_tag = rnode->indicator; 7689 7690 xmit->dif = 0; 7691 xmit->ct = SLI4_ELS_REQUEST64_CONTEXT_RPI; 7692 xmit->bs = 0; 7693 7694 xmit->command = SLI4_WQE_XMIT_SEQUENCE64; 7695 xmit->class = SLI4_ELS_REQUEST64_CLASS_3; 7696 xmit->pu = 0; 7697 xmit->timer = timeout; 7698 7699 xmit->abort_tag = 0; 7700 xmit->request_tag = tag; 7701 xmit->remote_xid = ox_id; 7702 7703 xmit->iod = SLI4_ELS_REQUEST64_DIR_READ; 7704 7705 if (rnode->node_group) { 7706 xmit->hlm = TRUE; 7707 xmit->remote_n_port_id = rnode->fc_id & 0x00ffffff; 7708 } 7709 7710 xmit->cmd_type = SLI4_CMD_XMIT_SEQUENCE64_WQE; 7711 7712 xmit->len_loc = 2; 7713 7714 xmit->cq_id = 0xFFFF; 7715 7716 return 0; 7717 } 7718 7719 /** 7720 * @ingroup sli_fc 7721 * @brief Write a REQUEUE_XRI_WQE work queue entry. 7722 * 7723 * @param sli4 SLI context. 7724 * @param buf Destination buffer for the WQE. 7725 * @param size Buffer size, in bytes. 7726 * @param xri XRI for this exchange. 7727 * @param tag IO tag value. 7728 * @param cq_id The id of the completion queue where the WQE response is sent. 7729 * 7730 * @return Returns 0 on success, or a non-zero value on failure. 7731 */ 7732 int32_t 7733 sli_requeue_xri_wqe(sli4_t *sli4, void *buf, size_t size, uint16_t xri, uint16_t tag, uint16_t cq_id) 7734 { 7735 sli4_requeue_xri_wqe_t *requeue = buf; 7736 7737 ocs_memset(buf, 0, size); 7738 7739 requeue->command = SLI4_WQE_REQUEUE_XRI; 7740 requeue->xri_tag = xri; 7741 requeue->request_tag = tag; 7742 requeue->xc = 1; 7743 requeue->qosd = 1; 7744 requeue->cq_id = cq_id; 7745 requeue->cmd_type = SLI4_CMD_REQUEUE_XRI_WQE; 7746 return 0; 7747 } 7748 7749 int32_t 7750 sli_xmit_bcast64_wqe(sli4_t *sli4, void *buf, size_t size, ocs_dma_t *payload, 7751 uint32_t payload_len, uint8_t timeout, uint16_t xri, uint16_t tag, 7752 uint16_t cq_id, ocs_remote_node_t *rnode, 7753 uint8_t r_ctl, uint8_t type, uint8_t df_ctl) 7754 { 7755 sli4_xmit_bcast64_wqe_t *bcast = buf; 7756 7757 /* Command requires a temporary RPI (i.e. unused remote node) */ 7758 if (rnode->attached) { 7759 ocs_log_test(sli4->os, "remote node %d in use\n", rnode->indicator); 7760 return -1; 7761 } 7762 7763 ocs_memset(buf, 0, size); 7764 7765 bcast->dbde = TRUE; 7766 bcast->sequence_payload.bde_type = SLI4_BDE_TYPE_BDE_64; 7767 bcast->sequence_payload.buffer_length = payload_len; 7768 bcast->sequence_payload.u.data.buffer_address_low = ocs_addr32_lo(payload->phys); 7769 bcast->sequence_payload.u.data.buffer_address_high = ocs_addr32_hi(payload->phys); 7770 7771 bcast->sequence_payload_length = payload_len; 7772 7773 bcast->df_ctl = df_ctl; 7774 bcast->type = type; 7775 bcast->r_ctl = r_ctl; 7776 7777 bcast->xri_tag = xri; 7778 7779 bcast->ct = SLI4_ELS_REQUEST64_CONTEXT_VPI; 7780 bcast->context_tag = rnode->sport->indicator; 7781 7782 bcast->class = SLI4_ELS_REQUEST64_CLASS_3; 7783 7784 bcast->command = SLI4_WQE_XMIT_BCAST64; 7785 7786 bcast->timer = timeout; 7787 7788 bcast->request_tag = tag; 7789 7790 bcast->temporary_rpi = rnode->indicator; 7791 7792 bcast->len_loc = 0x1; 7793 7794 bcast->iod = SLI4_ELS_REQUEST64_DIR_WRITE; 7795 7796 bcast->cmd_type = SLI4_CMD_XMIT_BCAST64_WQE; 7797 7798 bcast->cq_id = cq_id; 7799 7800 return 0; 7801 } 7802 7803 /** 7804 * @ingroup sli_fc 7805 * @brief Write an XMIT_BLS_RSP64_WQE work queue entry. 7806 * 7807 * @param sli4 SLI context. 7808 * @param buf Destination buffer for the WQE. 7809 * @param size Buffer size, in bytes. 7810 * @param payload Contents of the BLS payload to be sent. 7811 * @param xri XRI for this exchange. 7812 * @param tag IO tag value. 7813 * @param cq_id The id of the completion queue where the WQE response is sent. 7814 * @param rnode Destination of request (that is, the remote node). 7815 * @param s_id Source ID to use in the response. If UINT32_MAX, use SLI Port's ID. 7816 * 7817 * @return Returns 0 on success, or a non-zero value on failure. 7818 */ 7819 int32_t 7820 sli_xmit_bls_rsp64_wqe(sli4_t *sli4, void *buf, size_t size, sli_bls_payload_t *payload, 7821 uint16_t xri, uint16_t tag, uint16_t cq_id, ocs_remote_node_t *rnode, uint32_t s_id) 7822 { 7823 sli4_xmit_bls_rsp_wqe_t *bls = buf; 7824 7825 /* 7826 * Callers can either specify RPI or S_ID, but not both 7827 */ 7828 if (rnode->attached && (s_id != UINT32_MAX)) { 7829 ocs_log_test(sli4->os, "S_ID specified for attached remote node %d\n", 7830 rnode->indicator); 7831 return -1; 7832 } 7833 7834 ocs_memset(buf, 0, size); 7835 7836 if (SLI_BLS_ACC == payload->type) { 7837 bls->payload_word0 = (payload->u.acc.seq_id_last << 16) | 7838 (payload->u.acc.seq_id_validity << 24); 7839 bls->high_seq_cnt = payload->u.acc.high_seq_cnt; 7840 bls->low_seq_cnt = payload->u.acc.low_seq_cnt; 7841 } else if (SLI_BLS_RJT == payload->type) { 7842 bls->payload_word0 = *((uint32_t *)&payload->u.rjt); 7843 bls->ar = TRUE; 7844 } else { 7845 ocs_log_test(sli4->os, "bad BLS type %#x\n", 7846 payload->type); 7847 return -1; 7848 } 7849 7850 bls->ox_id = payload->ox_id; 7851 bls->rx_id = payload->rx_id; 7852 7853 if (rnode->attached) { 7854 bls->ct = SLI4_ELS_REQUEST64_CONTEXT_RPI; 7855 bls->context_tag = rnode->indicator; 7856 } else { 7857 bls->ct = SLI4_ELS_REQUEST64_CONTEXT_VPI; 7858 bls->context_tag = rnode->sport->indicator; 7859 7860 if (UINT32_MAX != s_id) { 7861 bls->local_n_port_id = s_id & 0x00ffffff; 7862 } else { 7863 bls->local_n_port_id = rnode->sport->fc_id & 0x00ffffff; 7864 } 7865 bls->remote_id = rnode->fc_id & 0x00ffffff; 7866 7867 bls->temporary_rpi = rnode->indicator; 7868 } 7869 7870 bls->xri_tag = xri; 7871 7872 bls->class = SLI4_ELS_REQUEST64_CLASS_3; 7873 7874 bls->command = SLI4_WQE_XMIT_BLS_RSP; 7875 7876 bls->request_tag = tag; 7877 7878 bls->qosd = TRUE; 7879 7880 if (rnode->node_group) { 7881 bls->hlm = TRUE; 7882 bls->remote_id = rnode->fc_id & 0x00ffffff; 7883 } 7884 7885 bls->cq_id = cq_id; 7886 7887 bls->cmd_type = SLI4_CMD_XMIT_BLS_RSP64_WQE; 7888 7889 return 0; 7890 } 7891 7892 /** 7893 * @ingroup sli_fc 7894 * @brief Write a XMIT_ELS_RSP64_WQE work queue entry. 7895 * 7896 * @param sli4 SLI context. 7897 * @param buf Destination buffer for the WQE. 7898 * @param size Buffer size, in bytes. 7899 * @param rsp DMA memory for the ELS response. 7900 * @param rsp_len Length of ELS response, in bytes. 7901 * @param xri XRI for this exchange. 7902 * @param tag IO tag value. 7903 * @param cq_id The id of the completion queue where the WQE response is sent. 7904 * @param ox_id OX_ID of the exchange containing the request. 7905 * @param rnode Destination of the ELS response (that is, the remote node). 7906 * @param flags Optional attributes, including: 7907 * - SLI4_IO_CONTINUATION - IO is already active. 7908 * @param s_id S_ID used for special responses. 7909 * 7910 * @return Returns 0 on success, or a non-zero value on failure. 7911 */ 7912 int32_t 7913 sli_xmit_els_rsp64_wqe(sli4_t *sli4, void *buf, size_t size, ocs_dma_t *rsp, 7914 uint32_t rsp_len, uint16_t xri, uint16_t tag, uint16_t cq_id, 7915 uint16_t ox_id, ocs_remote_node_t *rnode, uint32_t flags, uint32_t s_id) 7916 { 7917 sli4_xmit_els_rsp64_wqe_t *els = buf; 7918 7919 ocs_memset(buf, 0, size); 7920 7921 if (sli4->config.sgl_pre_registered) { 7922 els->dbde = TRUE; 7923 } else { 7924 els->xbl = TRUE; 7925 } 7926 7927 els->els_response_payload.bde_type = SLI4_BDE_TYPE_BDE_64; 7928 els->els_response_payload.buffer_length = rsp_len; 7929 els->els_response_payload.u.data.buffer_address_low = ocs_addr32_lo(rsp->phys); 7930 els->els_response_payload.u.data.buffer_address_high = ocs_addr32_hi(rsp->phys); 7931 7932 els->els_response_payload_length = rsp_len; 7933 7934 els->xri_tag = xri; 7935 7936 els->class = SLI4_ELS_REQUEST64_CLASS_3; 7937 7938 els->command = SLI4_WQE_ELS_RSP64; 7939 7940 els->request_tag = tag; 7941 7942 els->ox_id = ox_id; 7943 7944 els->iod = SLI4_ELS_REQUEST64_DIR_WRITE; 7945 7946 els->qosd = TRUE; 7947 7948 if (flags & SLI4_IO_CONTINUATION) { 7949 els->xc = TRUE; 7950 } 7951 7952 if (rnode->attached) { 7953 els->ct = SLI4_ELS_REQUEST64_CONTEXT_RPI; 7954 els->context_tag = rnode->indicator; 7955 } else { 7956 els->ct = SLI4_ELS_REQUEST64_CONTEXT_VPI; 7957 els->context_tag = rnode->sport->indicator; 7958 els->remote_id = rnode->fc_id & 0x00ffffff; 7959 els->temporary_rpi = rnode->indicator; 7960 if (UINT32_MAX != s_id) { 7961 els->sp = TRUE; 7962 els->s_id = s_id & 0x00ffffff; 7963 } 7964 } 7965 7966 if (rnode->node_group) { 7967 els->hlm = TRUE; 7968 els->remote_id = rnode->fc_id & 0x00ffffff; 7969 } 7970 7971 els->cmd_type = SLI4_ELS_REQUEST64_CMD_GEN; 7972 7973 els->cq_id = cq_id; 7974 7975 return 0; 7976 } 7977 7978 /** 7979 * @ingroup sli_fc 7980 * @brief Process an asynchronous Link State event entry. 7981 * 7982 * @par Description 7983 * Parses Asynchronous Completion Queue Entry (ACQE), 7984 * creates an abstracted event, and calls registered callback functions. 7985 * 7986 * @param sli4 SLI context. 7987 * @param acqe Pointer to the ACQE. 7988 * 7989 * @return Returns 0 on success, or a non-zero value on failure. 7990 */ 7991 int32_t 7992 sli_fc_process_link_state(sli4_t *sli4, void *acqe) 7993 { 7994 sli4_link_state_t *link_state = acqe; 7995 sli4_link_event_t event = { 0 }; 7996 int32_t rc = 0; 7997 7998 if (!sli4->link) { 7999 /* bail if there is no callback */ 8000 return 0; 8001 } 8002 8003 if (SLI4_LINK_TYPE_ETHERNET == link_state->link_type) { 8004 event.topology = SLI_LINK_TOPO_NPORT; 8005 event.medium = SLI_LINK_MEDIUM_ETHERNET; 8006 } else { 8007 /* TODO is this supported for anything other than FCoE? */ 8008 ocs_log_test(sli4->os, "unsupported link type %#x\n", 8009 link_state->link_type); 8010 event.topology = SLI_LINK_TOPO_MAX; 8011 event.medium = SLI_LINK_MEDIUM_MAX; 8012 rc = -1; 8013 } 8014 8015 switch (link_state->port_link_status) { 8016 case SLI4_PORT_LINK_STATUS_PHYSICAL_DOWN: 8017 case SLI4_PORT_LINK_STATUS_LOGICAL_DOWN: 8018 event.status = SLI_LINK_STATUS_DOWN; 8019 break; 8020 case SLI4_PORT_LINK_STATUS_PHYSICAL_UP: 8021 case SLI4_PORT_LINK_STATUS_LOGICAL_UP: 8022 event.status = SLI_LINK_STATUS_UP; 8023 break; 8024 default: 8025 ocs_log_test(sli4->os, "unsupported link status %#x\n", 8026 link_state->port_link_status); 8027 event.status = SLI_LINK_STATUS_MAX; 8028 rc = -1; 8029 } 8030 8031 switch (link_state->port_speed) { 8032 case 0: 8033 event.speed = 0; 8034 break; 8035 case 1: 8036 event.speed = 10; 8037 break; 8038 case 2: 8039 event.speed = 100; 8040 break; 8041 case 3: 8042 event.speed = 1000; 8043 break; 8044 case 4: 8045 event.speed = 10000; 8046 break; 8047 case 5: 8048 event.speed = 20000; 8049 break; 8050 case 6: 8051 event.speed = 25000; 8052 break; 8053 case 7: 8054 event.speed = 40000; 8055 break; 8056 case 8: 8057 event.speed = 100000; 8058 break; 8059 default: 8060 ocs_log_test(sli4->os, "unsupported port_speed %#x\n", 8061 link_state->port_speed); 8062 rc = -1; 8063 } 8064 8065 sli4->link(sli4->link_arg, (void *)&event); 8066 8067 return rc; 8068 } 8069 8070 /** 8071 * @ingroup sli_fc 8072 * @brief Process an asynchronous Link Attention event entry. 8073 * 8074 * @par Description 8075 * Parses Asynchronous Completion Queue Entry (ACQE), 8076 * creates an abstracted event, and calls the registered callback functions. 8077 * 8078 * @param sli4 SLI context. 8079 * @param acqe Pointer to the ACQE. 8080 * 8081 * @todo XXX all events return LINK_UP. 8082 * 8083 * @return Returns 0 on success, or a non-zero value on failure. 8084 */ 8085 int32_t 8086 sli_fc_process_link_attention(sli4_t *sli4, void *acqe) 8087 { 8088 sli4_link_attention_t *link_attn = acqe; 8089 sli4_link_event_t event = { 0 }; 8090 8091 ocs_log_debug(sli4->os, "link_number=%d attn_type=%#x topology=%#x port_speed=%#x " 8092 "port_fault=%#x shared_link_status=%#x logical_link_speed=%#x " 8093 "event_tag=%#x\n", link_attn->link_number, link_attn->attn_type, 8094 link_attn->topology, link_attn->port_speed, link_attn->port_fault, 8095 link_attn->shared_link_status, link_attn->logical_link_speed, 8096 link_attn->event_tag); 8097 8098 if (!sli4->link) { 8099 return 0; 8100 } 8101 8102 event.medium = SLI_LINK_MEDIUM_FC; 8103 8104 switch (link_attn->attn_type) { 8105 case SLI4_LINK_ATTN_TYPE_LINK_UP: 8106 event.status = SLI_LINK_STATUS_UP; 8107 break; 8108 case SLI4_LINK_ATTN_TYPE_LINK_DOWN: 8109 event.status = SLI_LINK_STATUS_DOWN; 8110 break; 8111 case SLI4_LINK_ATTN_TYPE_NO_HARD_ALPA: 8112 ocs_log_debug(sli4->os, "attn_type: no hard alpa\n"); 8113 event.status = SLI_LINK_STATUS_NO_ALPA; 8114 break; 8115 default: 8116 ocs_log_test(sli4->os, "attn_type: unknown\n"); 8117 break; 8118 } 8119 8120 switch (link_attn->event_type) { 8121 case SLI4_FC_EVENT_LINK_ATTENTION: 8122 break; 8123 case SLI4_FC_EVENT_SHARED_LINK_ATTENTION: 8124 ocs_log_debug(sli4->os, "event_type: FC shared link event \n"); 8125 break; 8126 default: 8127 ocs_log_test(sli4->os, "event_type: unknown\n"); 8128 break; 8129 } 8130 8131 switch (link_attn->topology) { 8132 case SLI4_LINK_ATTN_P2P: 8133 event.topology = SLI_LINK_TOPO_NPORT; 8134 break; 8135 case SLI4_LINK_ATTN_FC_AL: 8136 event.topology = SLI_LINK_TOPO_LOOP; 8137 break; 8138 case SLI4_LINK_ATTN_INTERNAL_LOOPBACK: 8139 ocs_log_debug(sli4->os, "topology Internal loopback\n"); 8140 event.topology = SLI_LINK_TOPO_LOOPBACK_INTERNAL; 8141 break; 8142 case SLI4_LINK_ATTN_SERDES_LOOPBACK: 8143 ocs_log_debug(sli4->os, "topology serdes loopback\n"); 8144 event.topology = SLI_LINK_TOPO_LOOPBACK_EXTERNAL; 8145 break; 8146 default: 8147 ocs_log_test(sli4->os, "topology: unknown\n"); 8148 break; 8149 } 8150 8151 event.speed = link_attn->port_speed * 1000; 8152 8153 sli4->link(sli4->link_arg, (void *)&event); 8154 8155 return 0; 8156 } 8157 8158 /** 8159 * @ingroup sli_fc 8160 * @brief Parse an FC/FCoE work queue CQ entry. 8161 * 8162 * @param sli4 SLI context. 8163 * @param cq CQ to process. 8164 * @param cqe Pointer to the CQ entry. 8165 * @param etype CQ event type. 8166 * @param r_id Resource ID associated with this completion message (such as the IO tag). 8167 * 8168 * @return Returns 0 on success, or a non-zero value on failure. 8169 */ 8170 int32_t 8171 sli_fc_cqe_parse(sli4_t *sli4, sli4_queue_t *cq, uint8_t *cqe, sli4_qentry_e *etype, 8172 uint16_t *r_id) 8173 { 8174 uint8_t code = cqe[SLI4_CQE_CODE_OFFSET]; 8175 int32_t rc = -1; 8176 8177 switch (code) { 8178 case SLI4_CQE_CODE_WORK_REQUEST_COMPLETION: 8179 { 8180 sli4_fc_wcqe_t *wcqe = (void *)cqe; 8181 8182 *etype = SLI_QENTRY_WQ; 8183 *r_id = wcqe->request_tag; 8184 rc = wcqe->status; 8185 8186 /* Flag errors except for FCP_RSP_FAILURE */ 8187 if (rc && (rc != SLI4_FC_WCQE_STATUS_FCP_RSP_FAILURE)) { 8188 ocs_log_test(sli4->os, "WCQE: status=%#x hw_status=%#x tag=%#x w1=%#x w2=%#x xb=%d\n", 8189 wcqe->status, wcqe->hw_status, 8190 wcqe->request_tag, wcqe->wqe_specific_1, 8191 wcqe->wqe_specific_2, wcqe->xb); 8192 ocs_log_test(sli4->os, " %08X %08X %08X %08X\n", ((uint32_t*) cqe)[0], ((uint32_t*) cqe)[1], 8193 ((uint32_t*) cqe)[2], ((uint32_t*) cqe)[3]); 8194 } 8195 8196 /* TODO: need to pass additional status back out of here as well 8197 * as status (could overload rc as status/addlstatus are only 8 bits each) 8198 */ 8199 8200 break; 8201 } 8202 case SLI4_CQE_CODE_RQ_ASYNC: 8203 { 8204 sli4_fc_async_rcqe_t *rcqe = (void *)cqe; 8205 8206 *etype = SLI_QENTRY_RQ; 8207 *r_id = rcqe->rq_id; 8208 rc = rcqe->status; 8209 break; 8210 } 8211 case SLI4_CQE_CODE_RQ_ASYNC_V1: 8212 { 8213 sli4_fc_async_rcqe_v1_t *rcqe = (void *)cqe; 8214 8215 *etype = SLI_QENTRY_RQ; 8216 *r_id = rcqe->rq_id; 8217 rc = rcqe->status; 8218 break; 8219 } 8220 case SLI4_CQE_CODE_OPTIMIZED_WRITE_CMD: 8221 { 8222 sli4_fc_optimized_write_cmd_cqe_t *optcqe = (void *)cqe; 8223 8224 *etype = SLI_QENTRY_OPT_WRITE_CMD; 8225 *r_id = optcqe->rq_id; 8226 rc = optcqe->status; 8227 break; 8228 } 8229 case SLI4_CQE_CODE_OPTIMIZED_WRITE_DATA: 8230 { 8231 sli4_fc_optimized_write_data_cqe_t *dcqe = (void *)cqe; 8232 8233 *etype = SLI_QENTRY_OPT_WRITE_DATA; 8234 *r_id = dcqe->xri; 8235 rc = dcqe->status; 8236 8237 /* Flag errors */ 8238 if (rc != SLI4_FC_WCQE_STATUS_SUCCESS) { 8239 ocs_log_test(sli4->os, "Optimized DATA CQE: status=%#x hw_status=%#x xri=%#x dpl=%#x w3=%#x xb=%d\n", 8240 dcqe->status, dcqe->hw_status, 8241 dcqe->xri, dcqe->total_data_placed, 8242 ((uint32_t*) cqe)[3], dcqe->xb); 8243 } 8244 break; 8245 } 8246 case SLI4_CQE_CODE_RQ_COALESCING: 8247 { 8248 sli4_fc_coalescing_rcqe_t *rcqe = (void *)cqe; 8249 8250 *etype = SLI_QENTRY_RQ; 8251 *r_id = rcqe->rq_id; 8252 rc = rcqe->status; 8253 break; 8254 } 8255 case SLI4_CQE_CODE_XRI_ABORTED: 8256 { 8257 sli4_fc_xri_aborted_cqe_t *xa = (void *)cqe; 8258 8259 *etype = SLI_QENTRY_XABT; 8260 *r_id = xa->xri; 8261 rc = 0; 8262 break; 8263 } 8264 case SLI4_CQE_CODE_RELEASE_WQE: { 8265 sli4_fc_wqec_t *wqec = (void*) cqe; 8266 8267 *etype = SLI_QENTRY_WQ_RELEASE; 8268 *r_id = wqec->wq_id; 8269 rc = 0; 8270 break; 8271 } 8272 default: 8273 ocs_log_test(sli4->os, "CQE completion code %d not handled\n", code); 8274 *etype = SLI_QENTRY_MAX; 8275 *r_id = UINT16_MAX; 8276 } 8277 8278 return rc; 8279 } 8280 8281 /** 8282 * @ingroup sli_fc 8283 * @brief Return the ELS/CT response length. 8284 * 8285 * @param sli4 SLI context. 8286 * @param cqe Pointer to the CQ entry. 8287 * 8288 * @return Returns the length, in bytes. 8289 */ 8290 uint32_t 8291 sli_fc_response_length(sli4_t *sli4, uint8_t *cqe) 8292 { 8293 sli4_fc_wcqe_t *wcqe = (void *)cqe; 8294 8295 return wcqe->wqe_specific_1; 8296 } 8297 8298 /** 8299 * @ingroup sli_fc 8300 * @brief Return the FCP IO length. 8301 * 8302 * @param sli4 SLI context. 8303 * @param cqe Pointer to the CQ entry. 8304 * 8305 * @return Returns the length, in bytes. 8306 */ 8307 uint32_t 8308 sli_fc_io_length(sli4_t *sli4, uint8_t *cqe) 8309 { 8310 sli4_fc_wcqe_t *wcqe = (void *)cqe; 8311 8312 return wcqe->wqe_specific_1; 8313 } 8314 8315 /** 8316 * @ingroup sli_fc 8317 * @brief Retrieve the D_ID from the completion. 8318 * 8319 * @param sli4 SLI context. 8320 * @param cqe Pointer to the CQ entry. 8321 * @param d_id Pointer where the D_ID is written. 8322 * 8323 * @return Returns 0 on success, or a non-zero value on failure. 8324 */ 8325 int32_t 8326 sli_fc_els_did(sli4_t *sli4, uint8_t *cqe, uint32_t *d_id) 8327 { 8328 sli4_fc_wcqe_t *wcqe = (void *)cqe; 8329 8330 *d_id = 0; 8331 8332 if (wcqe->status) { 8333 return -1; 8334 } else { 8335 *d_id = wcqe->wqe_specific_2 & 0x00ffffff; 8336 return 0; 8337 } 8338 } 8339 8340 uint32_t 8341 sli_fc_ext_status(sli4_t *sli4, uint8_t *cqe) 8342 { 8343 sli4_fc_wcqe_t *wcqe = (void *)cqe; 8344 uint32_t mask; 8345 8346 switch (wcqe->status) { 8347 case SLI4_FC_WCQE_STATUS_FCP_RSP_FAILURE: 8348 mask = UINT32_MAX; 8349 break; 8350 case SLI4_FC_WCQE_STATUS_LOCAL_REJECT: 8351 case SLI4_FC_WCQE_STATUS_CMD_REJECT: 8352 mask = 0xff; 8353 break; 8354 case SLI4_FC_WCQE_STATUS_NPORT_RJT: 8355 case SLI4_FC_WCQE_STATUS_FABRIC_RJT: 8356 case SLI4_FC_WCQE_STATUS_NPORT_BSY: 8357 case SLI4_FC_WCQE_STATUS_FABRIC_BSY: 8358 case SLI4_FC_WCQE_STATUS_LS_RJT: 8359 mask = UINT32_MAX; 8360 break; 8361 case SLI4_FC_WCQE_STATUS_DI_ERROR: 8362 mask = UINT32_MAX; 8363 break; 8364 default: 8365 mask = 0; 8366 } 8367 8368 return wcqe->wqe_specific_2 & mask; 8369 } 8370 8371 /** 8372 * @ingroup sli_fc 8373 * @brief Retrieve the RQ index from the completion. 8374 * 8375 * @param sli4 SLI context. 8376 * @param cqe Pointer to the CQ entry. 8377 * @param rq_id Pointer where the rq_id is written. 8378 * @param index Pointer where the index is written. 8379 * 8380 * @return Returns 0 on success, or a non-zero value on failure. 8381 */ 8382 int32_t 8383 sli_fc_rqe_rqid_and_index(sli4_t *sli4, uint8_t *cqe, uint16_t *rq_id, uint32_t *index) 8384 { 8385 sli4_fc_async_rcqe_t *rcqe = (void *)cqe; 8386 sli4_fc_async_rcqe_v1_t *rcqe_v1 = (void *)cqe; 8387 int32_t rc = -1; 8388 uint8_t code = 0; 8389 8390 *rq_id = 0; 8391 *index = UINT32_MAX; 8392 8393 code = cqe[SLI4_CQE_CODE_OFFSET]; 8394 8395 if (code == SLI4_CQE_CODE_RQ_ASYNC) { 8396 *rq_id = rcqe->rq_id; 8397 if (SLI4_FC_ASYNC_RQ_SUCCESS == rcqe->status) { 8398 *index = rcqe->rq_element_index; 8399 rc = 0; 8400 } else { 8401 *index = rcqe->rq_element_index; 8402 rc = rcqe->status; 8403 ocs_log_test(sli4->os, "status=%02x (%s) rq_id=%d, index=%x pdpl=%x sof=%02x eof=%02x hdpl=%x\n", 8404 rcqe->status, sli_fc_get_status_string(rcqe->status), rcqe->rq_id, 8405 rcqe->rq_element_index, rcqe->payload_data_placement_length, rcqe->sof_byte, 8406 rcqe->eof_byte, rcqe->header_data_placement_length); 8407 } 8408 } else if (code == SLI4_CQE_CODE_RQ_ASYNC_V1) { 8409 *rq_id = rcqe_v1->rq_id; 8410 if (SLI4_FC_ASYNC_RQ_SUCCESS == rcqe_v1->status) { 8411 *index = rcqe_v1->rq_element_index; 8412 rc = 0; 8413 } else { 8414 *index = rcqe_v1->rq_element_index; 8415 rc = rcqe_v1->status; 8416 ocs_log_test(sli4->os, "status=%02x (%s) rq_id=%d, index=%x pdpl=%x sof=%02x eof=%02x hdpl=%x\n", 8417 rcqe_v1->status, sli_fc_get_status_string(rcqe_v1->status), 8418 rcqe_v1->rq_id, rcqe_v1->rq_element_index, 8419 rcqe_v1->payload_data_placement_length, rcqe_v1->sof_byte, 8420 rcqe_v1->eof_byte, rcqe_v1->header_data_placement_length); 8421 } 8422 } else if (code == SLI4_CQE_CODE_OPTIMIZED_WRITE_CMD) { 8423 sli4_fc_optimized_write_cmd_cqe_t *optcqe = (void *)cqe; 8424 8425 *rq_id = optcqe->rq_id; 8426 if (SLI4_FC_ASYNC_RQ_SUCCESS == optcqe->status) { 8427 *index = optcqe->rq_element_index; 8428 rc = 0; 8429 } else { 8430 *index = optcqe->rq_element_index; 8431 rc = optcqe->status; 8432 ocs_log_test(sli4->os, "status=%02x (%s) rq_id=%d, index=%x pdpl=%x hdpl=%x oox=%d agxr=%d xri=0x%x rpi=0x%x\n", 8433 optcqe->status, sli_fc_get_status_string(optcqe->status), optcqe->rq_id, 8434 optcqe->rq_element_index, optcqe->payload_data_placement_length, 8435 optcqe->header_data_placement_length, optcqe->oox, optcqe->agxr, optcqe->xri, 8436 optcqe->rpi); 8437 } 8438 } else if (code == SLI4_CQE_CODE_RQ_COALESCING) { 8439 sli4_fc_coalescing_rcqe_t *rcqe = (void *)cqe; 8440 8441 *rq_id = rcqe->rq_id; 8442 if (SLI4_FC_COALESCE_RQ_SUCCESS == rcqe->status) { 8443 *index = rcqe->rq_element_index; 8444 rc = 0; 8445 } else { 8446 *index = UINT32_MAX; 8447 rc = rcqe->status; 8448 8449 ocs_log_test(sli4->os, "status=%02x (%s) rq_id=%d, index=%x rq_id=%#x sdpl=%x\n", 8450 rcqe->status, sli_fc_get_status_string(rcqe->status), rcqe->rq_id, 8451 rcqe->rq_element_index, rcqe->rq_id, rcqe->sequence_reporting_placement_length); 8452 } 8453 } else { 8454 *index = UINT32_MAX; 8455 8456 rc = rcqe->status; 8457 8458 ocs_log_debug(sli4->os, "status=%02x rq_id=%d, index=%x pdpl=%x sof=%02x eof=%02x hdpl=%x\n", 8459 rcqe->status, rcqe->rq_id, rcqe->rq_element_index, rcqe->payload_data_placement_length, 8460 rcqe->sof_byte, rcqe->eof_byte, rcqe->header_data_placement_length); 8461 } 8462 8463 return rc; 8464 } 8465 8466 /** 8467 * @ingroup sli_fc 8468 * @brief Process an asynchronous FCoE event entry. 8469 * 8470 * @par Description 8471 * Parses Asynchronous Completion Queue Entry (ACQE), 8472 * creates an abstracted event, and calls the registered callback functions. 8473 * 8474 * @param sli4 SLI context. 8475 * @param acqe Pointer to the ACQE. 8476 * 8477 * @return Returns 0 on success, or a non-zero value on failure. 8478 */ 8479 int32_t 8480 sli_fc_process_fcoe(sli4_t *sli4, void *acqe) 8481 { 8482 sli4_fcoe_fip_t *fcoe = acqe; 8483 sli4_fip_event_t event = { 0 }; 8484 uint32_t mask = UINT32_MAX; 8485 8486 ocs_log_debug(sli4->os, "ACQE FCoE FIP type=%02x count=%d tag=%#x\n", 8487 fcoe->event_type, 8488 fcoe->fcf_count, 8489 fcoe->event_tag); 8490 8491 if (!sli4->fip) { 8492 return 0; 8493 } 8494 8495 event.type = fcoe->event_type; 8496 event.index = UINT32_MAX; 8497 8498 switch (fcoe->event_type) { 8499 case SLI4_FCOE_FIP_FCF_DISCOVERED: 8500 ocs_log_debug(sli4->os, "FCF Discovered index=%d\n", fcoe->event_information); 8501 break; 8502 case SLI4_FCOE_FIP_FCF_TABLE_FULL: 8503 ocs_log_debug(sli4->os, "FCF Table Full\n"); 8504 mask = 0; 8505 break; 8506 case SLI4_FCOE_FIP_FCF_DEAD: 8507 ocs_log_debug(sli4->os, "FCF Dead/Gone index=%d\n", fcoe->event_information); 8508 break; 8509 case SLI4_FCOE_FIP_FCF_CLEAR_VLINK: 8510 mask = UINT16_MAX; 8511 ocs_log_debug(sli4->os, "Clear VLINK Received VPI=%#x\n", fcoe->event_information & mask); 8512 break; 8513 case SLI4_FCOE_FIP_FCF_MODIFIED: 8514 ocs_log_debug(sli4->os, "FCF Modified\n"); 8515 break; 8516 default: 8517 ocs_log_test(sli4->os, "bad FCoE type %#x", fcoe->event_type); 8518 mask = 0; 8519 } 8520 8521 if (mask != 0) { 8522 event.index = fcoe->event_information & mask; 8523 } 8524 8525 sli4->fip(sli4->fip_arg, &event); 8526 8527 return 0; 8528 } 8529 8530 /** 8531 * @ingroup sli_fc 8532 * @brief Allocate a receive queue. 8533 * 8534 * @par Description 8535 * Allocates DMA memory and configures the requested queue type. 8536 * 8537 * @param sli4 SLI context. 8538 * @param q Pointer to the queue object for the header. 8539 * @param n_entries Number of entries to allocate. 8540 * @param buffer_size buffer size for the queue. 8541 * @param cq Associated CQ. 8542 * @param ulp The ULP to bind 8543 * @param is_hdr Used to validate the rq_id and set the type of queue 8544 * 8545 * @return Returns 0 on success, or -1 on failure. 8546 */ 8547 int32_t 8548 sli_fc_rq_alloc(sli4_t *sli4, sli4_queue_t *q, 8549 uint32_t n_entries, uint32_t buffer_size, 8550 sli4_queue_t *cq, uint16_t ulp, uint8_t is_hdr) 8551 { 8552 int32_t (*rq_create)(sli4_t *, void *, size_t, ocs_dma_t *, uint16_t, uint16_t, uint16_t); 8553 8554 if ((sli4 == NULL) || (q == NULL)) { 8555 void *os = sli4 != NULL ? sli4->os : NULL; 8556 8557 ocs_log_err(os, "bad parameter sli4=%p q=%p\n", sli4, q); 8558 return -1; 8559 } 8560 8561 if (__sli_queue_init(sli4, q, SLI_QTYPE_RQ, SLI4_FCOE_RQE_SIZE, 8562 n_entries, SLI_PAGE_SIZE)) { 8563 return -1; 8564 } 8565 8566 if (sli4->if_type == SLI4_IF_TYPE_BE3_SKH_PF) { 8567 rq_create = sli_cmd_fcoe_rq_create; 8568 } else { 8569 rq_create = sli_cmd_fcoe_rq_create_v1; 8570 } 8571 8572 if (rq_create(sli4, sli4->bmbx.virt, SLI4_BMBX_SIZE, &q->dma, 8573 cq->id, ulp, buffer_size)) { 8574 if (__sli_create_queue(sli4, q)) { 8575 ocs_dma_free(sli4->os, &q->dma); 8576 return -1; 8577 } 8578 if (is_hdr && q->id & 1) { 8579 ocs_log_test(sli4->os, "bad header RQ_ID %d\n", q->id); 8580 ocs_dma_free(sli4->os, &q->dma); 8581 return -1; 8582 } else if (!is_hdr && (q->id & 1) == 0) { 8583 ocs_log_test(sli4->os, "bad data RQ_ID %d\n", q->id); 8584 ocs_dma_free(sli4->os, &q->dma); 8585 return -1; 8586 } 8587 } else { 8588 return -1; 8589 } 8590 q->u.flag.is_hdr = is_hdr; 8591 if (SLI4_IF_TYPE_BE3_SKH_PF == sli4->if_type) { 8592 q->u.flag.rq_batch = TRUE; 8593 } 8594 return 0; 8595 } 8596 8597 /** 8598 * @ingroup sli_fc 8599 * @brief Allocate a receive queue set. 8600 * 8601 * @param sli4 SLI context. 8602 * @param num_rq_pairs to create 8603 * @param qs Pointers to the queue objects for both header and data. 8604 * Length of this arrays should be 2 * num_rq_pairs 8605 * @param base_cq_id. Assumes base_cq_id : (base_cq_id + num_rq_pairs) cqs as allotted. 8606 * @param n_entries number of entries in each RQ queue. 8607 * @param header_buffer_size 8608 * @param payload_buffer_size 8609 * @param ulp The ULP to bind 8610 * 8611 * @return Returns 0 on success, or -1 on failure. 8612 */ 8613 int32_t 8614 sli_fc_rq_set_alloc(sli4_t *sli4, uint32_t num_rq_pairs, 8615 sli4_queue_t *qs[], uint32_t base_cq_id, 8616 uint32_t n_entries, uint32_t header_buffer_size, 8617 uint32_t payload_buffer_size, uint16_t ulp) 8618 { 8619 uint32_t i, p, offset = 0; 8620 uint32_t payload_size, total_page_count = 0; 8621 uintptr_t addr; 8622 ocs_dma_t dma; 8623 sli4_res_common_create_queue_set_t *rsp = NULL; 8624 sli4_req_fcoe_rq_create_v2_t *req = NULL; 8625 8626 ocs_memset(&dma, 0, sizeof(dma)); 8627 8628 for (i = 0; i < (num_rq_pairs * 2); i++) { 8629 if (__sli_queue_init(sli4, qs[i], SLI_QTYPE_RQ, SLI4_FCOE_RQE_SIZE, 8630 n_entries, SLI_PAGE_SIZE)) { 8631 goto error; 8632 } 8633 } 8634 8635 total_page_count = sli_page_count(qs[0]->dma.size, SLI_PAGE_SIZE) * num_rq_pairs * 2; 8636 8637 /* Payload length must accommodate both request and response */ 8638 payload_size = max((sizeof(sli4_req_fcoe_rq_create_v1_t) + (8 * total_page_count)), 8639 sizeof(sli4_res_common_create_queue_set_t)); 8640 8641 if (ocs_dma_alloc(sli4->os, &dma, payload_size, SLI_PAGE_SIZE)) { 8642 ocs_log_err(sli4->os, "DMA allocation failed\n"); 8643 goto error; 8644 } 8645 ocs_memset(dma.virt, 0, payload_size); 8646 8647 if (sli_cmd_sli_config(sli4, sli4->bmbx.virt, SLI4_BMBX_SIZE, 8648 payload_size, &dma) == -1) { 8649 goto error; 8650 } 8651 req = (sli4_req_fcoe_rq_create_v2_t *)((uint8_t *)dma.virt); 8652 8653 /* Fill Header fields */ 8654 req->hdr.opcode = SLI4_OPC_FCOE_RQ_CREATE; 8655 req->hdr.subsystem = SLI4_SUBSYSTEM_FCFCOE; 8656 req->hdr.version = 2; 8657 req->hdr.request_length = sizeof(sli4_req_fcoe_rq_create_v2_t) - sizeof(sli4_req_hdr_t) 8658 + (8 * total_page_count); 8659 8660 /* Fill Payload fields */ 8661 req->dnb = TRUE; 8662 req->num_pages = sli_page_count(qs[0]->dma.size, SLI_PAGE_SIZE); 8663 req->rqe_count = qs[0]->dma.size / SLI4_FCOE_RQE_SIZE; 8664 req->rqe_size = SLI4_FCOE_RQE_SIZE_8; 8665 req->page_size = SLI4_FCOE_RQ_PAGE_SIZE_4096; 8666 req->rq_count = num_rq_pairs * 2; 8667 req->base_cq_id = base_cq_id; 8668 req->hdr_buffer_size = header_buffer_size; 8669 req->payload_buffer_size = payload_buffer_size; 8670 8671 for (i = 0; i < (num_rq_pairs * 2); i++) { 8672 for (p = 0, addr = qs[i]->dma.phys; p < req->num_pages; p++, addr += SLI_PAGE_SIZE) { 8673 req->page_physical_address[offset].low = ocs_addr32_lo(addr); 8674 req->page_physical_address[offset].high = ocs_addr32_hi(addr); 8675 offset++; 8676 } 8677 } 8678 8679 if (sli_bmbx_command(sli4)){ 8680 ocs_log_crit(sli4->os, "bootstrap mailbox write faild RQSet\n"); 8681 goto error; 8682 } 8683 8684 rsp = (void *)((uint8_t *)dma.virt); 8685 if (rsp->hdr.status) { 8686 ocs_log_err(sli4->os, "bad create RQSet status=%#x addl=%#x\n", 8687 rsp->hdr.status, rsp->hdr.additional_status); 8688 goto error; 8689 } else { 8690 for (i = 0; i < (num_rq_pairs * 2); i++) { 8691 qs[i]->id = i + rsp->q_id; 8692 if ((qs[i]->id & 1) == 0) { 8693 qs[i]->u.flag.is_hdr = TRUE; 8694 } else { 8695 qs[i]->u.flag.is_hdr = FALSE; 8696 } 8697 qs[i]->doorbell_offset = regmap[SLI4_REG_FCOE_RQ_DOORBELL][sli4->if_type].off; 8698 qs[i]->doorbell_rset = regmap[SLI4_REG_FCOE_RQ_DOORBELL][sli4->if_type].rset; 8699 } 8700 } 8701 8702 ocs_dma_free(sli4->os, &dma); 8703 8704 return 0; 8705 8706 error: 8707 for (i = 0; i < (num_rq_pairs * 2); i++) { 8708 if (qs[i]->dma.size) { 8709 ocs_dma_free(sli4->os, &qs[i]->dma); 8710 } 8711 } 8712 8713 if (dma.size) { 8714 ocs_dma_free(sli4->os, &dma); 8715 } 8716 8717 return -1; 8718 } 8719 8720 /** 8721 * @ingroup sli_fc 8722 * @brief Get the RPI resource requirements. 8723 * 8724 * @param sli4 SLI context. 8725 * @param n_rpi Number of RPIs desired. 8726 * 8727 * @return Returns the number of bytes needed. This value may be zero. 8728 */ 8729 uint32_t 8730 sli_fc_get_rpi_requirements(sli4_t *sli4, uint32_t n_rpi) 8731 { 8732 uint32_t bytes = 0; 8733 8734 /* Check if header templates needed */ 8735 if (sli4->config.hdr_template_req) { 8736 /* round up to a page */ 8737 bytes = SLI_ROUND_PAGE(n_rpi * SLI4_FCOE_HDR_TEMPLATE_SIZE); 8738 } 8739 8740 return bytes; 8741 } 8742 8743 /** 8744 * @ingroup sli_fc 8745 * @brief Return a text string corresponding to a CQE status value 8746 * 8747 * @param status Status value 8748 * 8749 * @return Returns corresponding string, otherwise "unknown" 8750 */ 8751 const char * 8752 sli_fc_get_status_string(uint32_t status) 8753 { 8754 static struct { 8755 uint32_t code; 8756 const char *label; 8757 } lookup[] = { 8758 {SLI4_FC_WCQE_STATUS_SUCCESS, "SUCCESS"}, 8759 {SLI4_FC_WCQE_STATUS_FCP_RSP_FAILURE, "FCP_RSP_FAILURE"}, 8760 {SLI4_FC_WCQE_STATUS_REMOTE_STOP, "REMOTE_STOP"}, 8761 {SLI4_FC_WCQE_STATUS_LOCAL_REJECT, "LOCAL_REJECT"}, 8762 {SLI4_FC_WCQE_STATUS_NPORT_RJT, "NPORT_RJT"}, 8763 {SLI4_FC_WCQE_STATUS_FABRIC_RJT, "FABRIC_RJT"}, 8764 {SLI4_FC_WCQE_STATUS_NPORT_BSY, "NPORT_BSY"}, 8765 {SLI4_FC_WCQE_STATUS_FABRIC_BSY, "FABRIC_BSY"}, 8766 {SLI4_FC_WCQE_STATUS_LS_RJT, "LS_RJT"}, 8767 {SLI4_FC_WCQE_STATUS_CMD_REJECT, "CMD_REJECT"}, 8768 {SLI4_FC_WCQE_STATUS_FCP_TGT_LENCHECK, "FCP_TGT_LENCHECK"}, 8769 {SLI4_FC_WCQE_STATUS_RQ_BUF_LEN_EXCEEDED, "BUF_LEN_EXCEEDED"}, 8770 {SLI4_FC_WCQE_STATUS_RQ_INSUFF_BUF_NEEDED, "RQ_INSUFF_BUF_NEEDED"}, 8771 {SLI4_FC_WCQE_STATUS_RQ_INSUFF_FRM_DISC, "RQ_INSUFF_FRM_DESC"}, 8772 {SLI4_FC_WCQE_STATUS_RQ_DMA_FAILURE, "RQ_DMA_FAILURE"}, 8773 {SLI4_FC_WCQE_STATUS_FCP_RSP_TRUNCATE, "FCP_RSP_TRUNCATE"}, 8774 {SLI4_FC_WCQE_STATUS_DI_ERROR, "DI_ERROR"}, 8775 {SLI4_FC_WCQE_STATUS_BA_RJT, "BA_RJT"}, 8776 {SLI4_FC_WCQE_STATUS_RQ_INSUFF_XRI_NEEDED, "RQ_INSUFF_XRI_NEEDED"}, 8777 {SLI4_FC_WCQE_STATUS_RQ_INSUFF_XRI_DISC, "INSUFF_XRI_DISC"}, 8778 {SLI4_FC_WCQE_STATUS_RX_ERROR_DETECT, "RX_ERROR_DETECT"}, 8779 {SLI4_FC_WCQE_STATUS_RX_ABORT_REQUEST, "RX_ABORT_REQUEST"}, 8780 }; 8781 uint32_t i; 8782 8783 for (i = 0; i < ARRAY_SIZE(lookup); i++) { 8784 if (status == lookup[i].code) { 8785 return lookup[i].label; 8786 } 8787 } 8788 return "unknown"; 8789 } 8790