1 /*- 2 * SPDX-License-Identifier: BSD-2-Clause-FreeBSD 3 * 4 * Copyright (c) 2013-2016 Qlogic Corporation 5 * All rights reserved. 6 * 7 * Redistribution and use in source and binary forms, with or without 8 * modification, are permitted provided that the following conditions 9 * are met: 10 * 11 * 1. Redistributions of source code must retain the above copyright 12 * notice, this list of conditions and the following disclaimer. 13 * 2. Redistributions in binary form must reproduce the above copyright 14 * notice, this list of conditions and the following disclaimer in the 15 * documentation and/or other materials provided with the distribution. 16 * 17 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" 18 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 19 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 20 * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE 21 * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR 22 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF 23 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS 24 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN 25 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) 26 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE 27 * POSSIBILITY OF SUCH DAMAGE. 28 */ 29 30 /* 31 * File: ql_hw.c 32 * Author : David C Somayajulu, Qlogic Corporation, Aliso Viejo, CA 92656. 33 * Content: Contains Hardware dependent functions 34 */ 35 36 #include <sys/cdefs.h> 37 __FBSDID("$FreeBSD$"); 38 39 #include "ql_os.h" 40 #include "ql_hw.h" 41 #include "ql_def.h" 42 #include "ql_inline.h" 43 #include "ql_ver.h" 44 #include "ql_glbl.h" 45 #include "ql_dbg.h" 46 #include "ql_minidump.h" 47 48 /* 49 * Static Functions 50 */ 51 52 static void qla_del_rcv_cntxt(qla_host_t *ha); 53 static int qla_init_rcv_cntxt(qla_host_t *ha); 54 static int qla_del_xmt_cntxt(qla_host_t *ha); 55 static int qla_init_xmt_cntxt(qla_host_t *ha); 56 static int qla_mbx_cmd(qla_host_t *ha, uint32_t *h_mbox, uint32_t n_hmbox, 57 uint32_t *fw_mbox, uint32_t n_fwmbox, uint32_t no_pause); 58 static int qla_config_intr_cntxt(qla_host_t *ha, uint32_t start_idx, 59 uint32_t num_intrs, uint32_t create); 60 static int qla_config_rss(qla_host_t *ha, uint16_t cntxt_id); 61 static int qla_config_intr_coalesce(qla_host_t *ha, uint16_t cntxt_id, 62 int tenable, int rcv); 63 static int qla_set_mac_rcv_mode(qla_host_t *ha, uint32_t mode); 64 static int qla_link_event_req(qla_host_t *ha, uint16_t cntxt_id); 65 66 static int qla_tx_tso(qla_host_t *ha, struct mbuf *mp, q80_tx_cmd_t *tx_cmd, 67 uint8_t *hdr); 68 static int qla_hw_add_all_mcast(qla_host_t *ha); 69 static int qla_add_rcv_rings(qla_host_t *ha, uint32_t sds_idx, uint32_t nsds); 70 71 static int qla_init_nic_func(qla_host_t *ha); 72 static int qla_stop_nic_func(qla_host_t *ha); 73 static int qla_query_fw_dcbx_caps(qla_host_t *ha); 74 static int qla_set_port_config(qla_host_t *ha, uint32_t cfg_bits); 75 static int qla_get_port_config(qla_host_t *ha, uint32_t *cfg_bits); 76 static int qla_set_cam_search_mode(qla_host_t *ha, uint32_t search_mode); 77 static int qla_get_cam_search_mode(qla_host_t *ha); 78 79 static void ql_minidump_free(qla_host_t *ha); 80 81 #ifdef QL_DBG 82 83 static void 84 qla_stop_pegs(qla_host_t *ha) 85 { 86 uint32_t val = 1; 87 88 ql_rdwr_indreg32(ha, Q8_CRB_PEG_0, &val, 0); 89 ql_rdwr_indreg32(ha, Q8_CRB_PEG_1, &val, 0); 90 ql_rdwr_indreg32(ha, Q8_CRB_PEG_2, &val, 0); 91 ql_rdwr_indreg32(ha, Q8_CRB_PEG_3, &val, 0); 92 ql_rdwr_indreg32(ha, Q8_CRB_PEG_4, &val, 0); 93 device_printf(ha->pci_dev, "%s PEGS HALTED!!!!!\n", __func__); 94 } 95 96 static int 97 qla_sysctl_stop_pegs(SYSCTL_HANDLER_ARGS) 98 { 99 int err, ret = 0; 100 qla_host_t *ha; 101 102 err = sysctl_handle_int(oidp, &ret, 0, req); 103 104 105 if (err || !req->newptr) 106 return (err); 107 108 if (ret == 1) { 109 ha = (qla_host_t *)arg1; 110 if (QLA_LOCK(ha, __func__, QLA_LOCK_DEFAULT_MS_TIMEOUT, 0) == 0) { 111 qla_stop_pegs(ha); 112 QLA_UNLOCK(ha, __func__); 113 } 114 } 115 116 return err; 117 } 118 #endif /* #ifdef QL_DBG */ 119 120 static int 121 qla_validate_set_port_cfg_bit(uint32_t bits) 122 { 123 if ((bits & 0xF) > 1) 124 return (-1); 125 126 if (((bits >> 4) & 0xF) > 2) 127 return (-1); 128 129 if (((bits >> 8) & 0xF) > 2) 130 return (-1); 131 132 return (0); 133 } 134 135 static int 136 qla_sysctl_port_cfg(SYSCTL_HANDLER_ARGS) 137 { 138 int err, ret = 0; 139 qla_host_t *ha; 140 uint32_t cfg_bits; 141 142 err = sysctl_handle_int(oidp, &ret, 0, req); 143 144 if (err || !req->newptr) 145 return (err); 146 147 ha = (qla_host_t *)arg1; 148 149 if ((qla_validate_set_port_cfg_bit((uint32_t)ret) == 0)) { 150 151 err = qla_get_port_config(ha, &cfg_bits); 152 153 if (err) 154 goto qla_sysctl_set_port_cfg_exit; 155 156 if (ret & 0x1) { 157 cfg_bits |= Q8_PORT_CFG_BITS_DCBX_ENABLE; 158 } else { 159 cfg_bits &= ~Q8_PORT_CFG_BITS_DCBX_ENABLE; 160 } 161 162 ret = ret >> 4; 163 cfg_bits &= ~Q8_PORT_CFG_BITS_PAUSE_CFG_MASK; 164 165 if ((ret & 0xF) == 0) { 166 cfg_bits |= Q8_PORT_CFG_BITS_PAUSE_DISABLED; 167 } else if ((ret & 0xF) == 1){ 168 cfg_bits |= Q8_PORT_CFG_BITS_PAUSE_STD; 169 } else { 170 cfg_bits |= Q8_PORT_CFG_BITS_PAUSE_PPM; 171 } 172 173 ret = ret >> 4; 174 cfg_bits &= ~Q8_PORT_CFG_BITS_STDPAUSE_DIR_MASK; 175 176 if (ret == 0) { 177 cfg_bits |= Q8_PORT_CFG_BITS_STDPAUSE_XMT_RCV; 178 } else if (ret == 1){ 179 cfg_bits |= Q8_PORT_CFG_BITS_STDPAUSE_XMT; 180 } else { 181 cfg_bits |= Q8_PORT_CFG_BITS_STDPAUSE_RCV; 182 } 183 184 if (QLA_LOCK(ha, __func__, QLA_LOCK_DEFAULT_MS_TIMEOUT, 0) == 0) { 185 err = qla_set_port_config(ha, cfg_bits); 186 QLA_UNLOCK(ha, __func__); 187 } else { 188 device_printf(ha->pci_dev, "%s: failed\n", __func__); 189 } 190 } else { 191 if (QLA_LOCK(ha, __func__, QLA_LOCK_DEFAULT_MS_TIMEOUT, 0) == 0) { 192 err = qla_get_port_config(ha, &cfg_bits); 193 QLA_UNLOCK(ha, __func__); 194 } else { 195 device_printf(ha->pci_dev, "%s: failed\n", __func__); 196 } 197 } 198 199 qla_sysctl_set_port_cfg_exit: 200 return err; 201 } 202 203 static int 204 qla_sysctl_set_cam_search_mode(SYSCTL_HANDLER_ARGS) 205 { 206 int err, ret = 0; 207 qla_host_t *ha; 208 209 err = sysctl_handle_int(oidp, &ret, 0, req); 210 211 if (err || !req->newptr) 212 return (err); 213 214 ha = (qla_host_t *)arg1; 215 216 if ((ret == Q8_HW_CONFIG_CAM_SEARCH_MODE_INTERNAL) || 217 (ret == Q8_HW_CONFIG_CAM_SEARCH_MODE_AUTO)) { 218 219 if (QLA_LOCK(ha, __func__, QLA_LOCK_DEFAULT_MS_TIMEOUT, 0) == 0) { 220 err = qla_set_cam_search_mode(ha, (uint32_t)ret); 221 QLA_UNLOCK(ha, __func__); 222 } else { 223 device_printf(ha->pci_dev, "%s: failed\n", __func__); 224 } 225 226 } else { 227 device_printf(ha->pci_dev, "%s: ret = %d\n", __func__, ret); 228 } 229 230 return (err); 231 } 232 233 static int 234 qla_sysctl_get_cam_search_mode(SYSCTL_HANDLER_ARGS) 235 { 236 int err, ret = 0; 237 qla_host_t *ha; 238 239 err = sysctl_handle_int(oidp, &ret, 0, req); 240 241 if (err || !req->newptr) 242 return (err); 243 244 ha = (qla_host_t *)arg1; 245 if (QLA_LOCK(ha, __func__, QLA_LOCK_DEFAULT_MS_TIMEOUT, 0) == 0) { 246 err = qla_get_cam_search_mode(ha); 247 QLA_UNLOCK(ha, __func__); 248 } else { 249 device_printf(ha->pci_dev, "%s: failed\n", __func__); 250 } 251 252 return (err); 253 } 254 255 static void 256 qlnx_add_hw_mac_stats_sysctls(qla_host_t *ha) 257 { 258 struct sysctl_ctx_list *ctx; 259 struct sysctl_oid_list *children; 260 struct sysctl_oid *ctx_oid; 261 262 ctx = device_get_sysctl_ctx(ha->pci_dev); 263 children = SYSCTL_CHILDREN(device_get_sysctl_tree(ha->pci_dev)); 264 265 ctx_oid = SYSCTL_ADD_NODE(ctx, children, OID_AUTO, "stats_hw_mac", 266 CTLFLAG_RD | CTLFLAG_MPSAFE, NULL, "stats_hw_mac"); 267 children = SYSCTL_CHILDREN(ctx_oid); 268 269 SYSCTL_ADD_QUAD(ctx, children, 270 OID_AUTO, "xmt_frames", 271 CTLFLAG_RD, &ha->hw.mac.xmt_frames, 272 "xmt_frames"); 273 274 SYSCTL_ADD_QUAD(ctx, children, 275 OID_AUTO, "xmt_bytes", 276 CTLFLAG_RD, &ha->hw.mac.xmt_bytes, 277 "xmt_frames"); 278 279 SYSCTL_ADD_QUAD(ctx, children, 280 OID_AUTO, "xmt_mcast_pkts", 281 CTLFLAG_RD, &ha->hw.mac.xmt_mcast_pkts, 282 "xmt_mcast_pkts"); 283 284 SYSCTL_ADD_QUAD(ctx, children, 285 OID_AUTO, "xmt_bcast_pkts", 286 CTLFLAG_RD, &ha->hw.mac.xmt_bcast_pkts, 287 "xmt_bcast_pkts"); 288 289 SYSCTL_ADD_QUAD(ctx, children, 290 OID_AUTO, "xmt_pause_frames", 291 CTLFLAG_RD, &ha->hw.mac.xmt_pause_frames, 292 "xmt_pause_frames"); 293 294 SYSCTL_ADD_QUAD(ctx, children, 295 OID_AUTO, "xmt_cntrl_pkts", 296 CTLFLAG_RD, &ha->hw.mac.xmt_cntrl_pkts, 297 "xmt_cntrl_pkts"); 298 299 SYSCTL_ADD_QUAD(ctx, children, 300 OID_AUTO, "xmt_pkt_lt_64bytes", 301 CTLFLAG_RD, &ha->hw.mac.xmt_pkt_lt_64bytes, 302 "xmt_pkt_lt_64bytes"); 303 304 SYSCTL_ADD_QUAD(ctx, children, 305 OID_AUTO, "xmt_pkt_lt_127bytes", 306 CTLFLAG_RD, &ha->hw.mac.xmt_pkt_lt_127bytes, 307 "xmt_pkt_lt_127bytes"); 308 309 SYSCTL_ADD_QUAD(ctx, children, 310 OID_AUTO, "xmt_pkt_lt_255bytes", 311 CTLFLAG_RD, &ha->hw.mac.xmt_pkt_lt_255bytes, 312 "xmt_pkt_lt_255bytes"); 313 314 SYSCTL_ADD_QUAD(ctx, children, 315 OID_AUTO, "xmt_pkt_lt_511bytes", 316 CTLFLAG_RD, &ha->hw.mac.xmt_pkt_lt_511bytes, 317 "xmt_pkt_lt_511bytes"); 318 319 SYSCTL_ADD_QUAD(ctx, children, 320 OID_AUTO, "xmt_pkt_lt_1023bytes", 321 CTLFLAG_RD, &ha->hw.mac.xmt_pkt_lt_1023bytes, 322 "xmt_pkt_lt_1023bytes"); 323 324 SYSCTL_ADD_QUAD(ctx, children, 325 OID_AUTO, "xmt_pkt_lt_1518bytes", 326 CTLFLAG_RD, &ha->hw.mac.xmt_pkt_lt_1518bytes, 327 "xmt_pkt_lt_1518bytes"); 328 329 SYSCTL_ADD_QUAD(ctx, children, 330 OID_AUTO, "xmt_pkt_gt_1518bytes", 331 CTLFLAG_RD, &ha->hw.mac.xmt_pkt_gt_1518bytes, 332 "xmt_pkt_gt_1518bytes"); 333 334 SYSCTL_ADD_QUAD(ctx, children, 335 OID_AUTO, "rcv_frames", 336 CTLFLAG_RD, &ha->hw.mac.rcv_frames, 337 "rcv_frames"); 338 339 SYSCTL_ADD_QUAD(ctx, children, 340 OID_AUTO, "rcv_bytes", 341 CTLFLAG_RD, &ha->hw.mac.rcv_bytes, 342 "rcv_bytes"); 343 344 SYSCTL_ADD_QUAD(ctx, children, 345 OID_AUTO, "rcv_mcast_pkts", 346 CTLFLAG_RD, &ha->hw.mac.rcv_mcast_pkts, 347 "rcv_mcast_pkts"); 348 349 SYSCTL_ADD_QUAD(ctx, children, 350 OID_AUTO, "rcv_bcast_pkts", 351 CTLFLAG_RD, &ha->hw.mac.rcv_bcast_pkts, 352 "rcv_bcast_pkts"); 353 354 SYSCTL_ADD_QUAD(ctx, children, 355 OID_AUTO, "rcv_pause_frames", 356 CTLFLAG_RD, &ha->hw.mac.rcv_pause_frames, 357 "rcv_pause_frames"); 358 359 SYSCTL_ADD_QUAD(ctx, children, 360 OID_AUTO, "rcv_cntrl_pkts", 361 CTLFLAG_RD, &ha->hw.mac.rcv_cntrl_pkts, 362 "rcv_cntrl_pkts"); 363 364 SYSCTL_ADD_QUAD(ctx, children, 365 OID_AUTO, "rcv_pkt_lt_64bytes", 366 CTLFLAG_RD, &ha->hw.mac.rcv_pkt_lt_64bytes, 367 "rcv_pkt_lt_64bytes"); 368 369 SYSCTL_ADD_QUAD(ctx, children, 370 OID_AUTO, "rcv_pkt_lt_127bytes", 371 CTLFLAG_RD, &ha->hw.mac.rcv_pkt_lt_127bytes, 372 "rcv_pkt_lt_127bytes"); 373 374 SYSCTL_ADD_QUAD(ctx, children, 375 OID_AUTO, "rcv_pkt_lt_255bytes", 376 CTLFLAG_RD, &ha->hw.mac.rcv_pkt_lt_255bytes, 377 "rcv_pkt_lt_255bytes"); 378 379 SYSCTL_ADD_QUAD(ctx, children, 380 OID_AUTO, "rcv_pkt_lt_511bytes", 381 CTLFLAG_RD, &ha->hw.mac.rcv_pkt_lt_511bytes, 382 "rcv_pkt_lt_511bytes"); 383 384 SYSCTL_ADD_QUAD(ctx, children, 385 OID_AUTO, "rcv_pkt_lt_1023bytes", 386 CTLFLAG_RD, &ha->hw.mac.rcv_pkt_lt_1023bytes, 387 "rcv_pkt_lt_1023bytes"); 388 389 SYSCTL_ADD_QUAD(ctx, children, 390 OID_AUTO, "rcv_pkt_lt_1518bytes", 391 CTLFLAG_RD, &ha->hw.mac.rcv_pkt_lt_1518bytes, 392 "rcv_pkt_lt_1518bytes"); 393 394 SYSCTL_ADD_QUAD(ctx, children, 395 OID_AUTO, "rcv_pkt_gt_1518bytes", 396 CTLFLAG_RD, &ha->hw.mac.rcv_pkt_gt_1518bytes, 397 "rcv_pkt_gt_1518bytes"); 398 399 SYSCTL_ADD_QUAD(ctx, children, 400 OID_AUTO, "rcv_len_error", 401 CTLFLAG_RD, &ha->hw.mac.rcv_len_error, 402 "rcv_len_error"); 403 404 SYSCTL_ADD_QUAD(ctx, children, 405 OID_AUTO, "rcv_len_small", 406 CTLFLAG_RD, &ha->hw.mac.rcv_len_small, 407 "rcv_len_small"); 408 409 SYSCTL_ADD_QUAD(ctx, children, 410 OID_AUTO, "rcv_len_large", 411 CTLFLAG_RD, &ha->hw.mac.rcv_len_large, 412 "rcv_len_large"); 413 414 SYSCTL_ADD_QUAD(ctx, children, 415 OID_AUTO, "rcv_jabber", 416 CTLFLAG_RD, &ha->hw.mac.rcv_jabber, 417 "rcv_jabber"); 418 419 SYSCTL_ADD_QUAD(ctx, children, 420 OID_AUTO, "rcv_dropped", 421 CTLFLAG_RD, &ha->hw.mac.rcv_dropped, 422 "rcv_dropped"); 423 424 SYSCTL_ADD_QUAD(ctx, children, 425 OID_AUTO, "fcs_error", 426 CTLFLAG_RD, &ha->hw.mac.fcs_error, 427 "fcs_error"); 428 429 SYSCTL_ADD_QUAD(ctx, children, 430 OID_AUTO, "align_error", 431 CTLFLAG_RD, &ha->hw.mac.align_error, 432 "align_error"); 433 434 SYSCTL_ADD_QUAD(ctx, children, 435 OID_AUTO, "eswitched_frames", 436 CTLFLAG_RD, &ha->hw.mac.eswitched_frames, 437 "eswitched_frames"); 438 439 SYSCTL_ADD_QUAD(ctx, children, 440 OID_AUTO, "eswitched_bytes", 441 CTLFLAG_RD, &ha->hw.mac.eswitched_bytes, 442 "eswitched_bytes"); 443 444 SYSCTL_ADD_QUAD(ctx, children, 445 OID_AUTO, "eswitched_mcast_frames", 446 CTLFLAG_RD, &ha->hw.mac.eswitched_mcast_frames, 447 "eswitched_mcast_frames"); 448 449 SYSCTL_ADD_QUAD(ctx, children, 450 OID_AUTO, "eswitched_bcast_frames", 451 CTLFLAG_RD, &ha->hw.mac.eswitched_bcast_frames, 452 "eswitched_bcast_frames"); 453 454 SYSCTL_ADD_QUAD(ctx, children, 455 OID_AUTO, "eswitched_ucast_frames", 456 CTLFLAG_RD, &ha->hw.mac.eswitched_ucast_frames, 457 "eswitched_ucast_frames"); 458 459 SYSCTL_ADD_QUAD(ctx, children, 460 OID_AUTO, "eswitched_err_free_frames", 461 CTLFLAG_RD, &ha->hw.mac.eswitched_err_free_frames, 462 "eswitched_err_free_frames"); 463 464 SYSCTL_ADD_QUAD(ctx, children, 465 OID_AUTO, "eswitched_err_free_bytes", 466 CTLFLAG_RD, &ha->hw.mac.eswitched_err_free_bytes, 467 "eswitched_err_free_bytes"); 468 469 return; 470 } 471 472 static void 473 qlnx_add_hw_rcv_stats_sysctls(qla_host_t *ha) 474 { 475 struct sysctl_ctx_list *ctx; 476 struct sysctl_oid_list *children; 477 struct sysctl_oid *ctx_oid; 478 479 ctx = device_get_sysctl_ctx(ha->pci_dev); 480 children = SYSCTL_CHILDREN(device_get_sysctl_tree(ha->pci_dev)); 481 482 ctx_oid = SYSCTL_ADD_NODE(ctx, children, OID_AUTO, "stats_hw_rcv", 483 CTLFLAG_RD | CTLFLAG_MPSAFE, NULL, "stats_hw_rcv"); 484 children = SYSCTL_CHILDREN(ctx_oid); 485 486 SYSCTL_ADD_QUAD(ctx, children, 487 OID_AUTO, "total_bytes", 488 CTLFLAG_RD, &ha->hw.rcv.total_bytes, 489 "total_bytes"); 490 491 SYSCTL_ADD_QUAD(ctx, children, 492 OID_AUTO, "total_pkts", 493 CTLFLAG_RD, &ha->hw.rcv.total_pkts, 494 "total_pkts"); 495 496 SYSCTL_ADD_QUAD(ctx, children, 497 OID_AUTO, "lro_pkt_count", 498 CTLFLAG_RD, &ha->hw.rcv.lro_pkt_count, 499 "lro_pkt_count"); 500 501 SYSCTL_ADD_QUAD(ctx, children, 502 OID_AUTO, "sw_pkt_count", 503 CTLFLAG_RD, &ha->hw.rcv.sw_pkt_count, 504 "sw_pkt_count"); 505 506 SYSCTL_ADD_QUAD(ctx, children, 507 OID_AUTO, "ip_chksum_err", 508 CTLFLAG_RD, &ha->hw.rcv.ip_chksum_err, 509 "ip_chksum_err"); 510 511 SYSCTL_ADD_QUAD(ctx, children, 512 OID_AUTO, "pkts_wo_acntxts", 513 CTLFLAG_RD, &ha->hw.rcv.pkts_wo_acntxts, 514 "pkts_wo_acntxts"); 515 516 SYSCTL_ADD_QUAD(ctx, children, 517 OID_AUTO, "pkts_dropped_no_sds_card", 518 CTLFLAG_RD, &ha->hw.rcv.pkts_dropped_no_sds_card, 519 "pkts_dropped_no_sds_card"); 520 521 SYSCTL_ADD_QUAD(ctx, children, 522 OID_AUTO, "pkts_dropped_no_sds_host", 523 CTLFLAG_RD, &ha->hw.rcv.pkts_dropped_no_sds_host, 524 "pkts_dropped_no_sds_host"); 525 526 SYSCTL_ADD_QUAD(ctx, children, 527 OID_AUTO, "oversized_pkts", 528 CTLFLAG_RD, &ha->hw.rcv.oversized_pkts, 529 "oversized_pkts"); 530 531 SYSCTL_ADD_QUAD(ctx, children, 532 OID_AUTO, "pkts_dropped_no_rds", 533 CTLFLAG_RD, &ha->hw.rcv.pkts_dropped_no_rds, 534 "pkts_dropped_no_rds"); 535 536 SYSCTL_ADD_QUAD(ctx, children, 537 OID_AUTO, "unxpctd_mcast_pkts", 538 CTLFLAG_RD, &ha->hw.rcv.unxpctd_mcast_pkts, 539 "unxpctd_mcast_pkts"); 540 541 SYSCTL_ADD_QUAD(ctx, children, 542 OID_AUTO, "re1_fbq_error", 543 CTLFLAG_RD, &ha->hw.rcv.re1_fbq_error, 544 "re1_fbq_error"); 545 546 SYSCTL_ADD_QUAD(ctx, children, 547 OID_AUTO, "invalid_mac_addr", 548 CTLFLAG_RD, &ha->hw.rcv.invalid_mac_addr, 549 "invalid_mac_addr"); 550 551 SYSCTL_ADD_QUAD(ctx, children, 552 OID_AUTO, "rds_prime_trys", 553 CTLFLAG_RD, &ha->hw.rcv.rds_prime_trys, 554 "rds_prime_trys"); 555 556 SYSCTL_ADD_QUAD(ctx, children, 557 OID_AUTO, "rds_prime_success", 558 CTLFLAG_RD, &ha->hw.rcv.rds_prime_success, 559 "rds_prime_success"); 560 561 SYSCTL_ADD_QUAD(ctx, children, 562 OID_AUTO, "lro_flows_added", 563 CTLFLAG_RD, &ha->hw.rcv.lro_flows_added, 564 "lro_flows_added"); 565 566 SYSCTL_ADD_QUAD(ctx, children, 567 OID_AUTO, "lro_flows_deleted", 568 CTLFLAG_RD, &ha->hw.rcv.lro_flows_deleted, 569 "lro_flows_deleted"); 570 571 SYSCTL_ADD_QUAD(ctx, children, 572 OID_AUTO, "lro_flows_active", 573 CTLFLAG_RD, &ha->hw.rcv.lro_flows_active, 574 "lro_flows_active"); 575 576 SYSCTL_ADD_QUAD(ctx, children, 577 OID_AUTO, "pkts_droped_unknown", 578 CTLFLAG_RD, &ha->hw.rcv.pkts_droped_unknown, 579 "pkts_droped_unknown"); 580 581 SYSCTL_ADD_QUAD(ctx, children, 582 OID_AUTO, "pkts_cnt_oversized", 583 CTLFLAG_RD, &ha->hw.rcv.pkts_cnt_oversized, 584 "pkts_cnt_oversized"); 585 586 return; 587 } 588 589 static void 590 qlnx_add_hw_xmt_stats_sysctls(qla_host_t *ha) 591 { 592 struct sysctl_ctx_list *ctx; 593 struct sysctl_oid_list *children; 594 struct sysctl_oid_list *node_children; 595 struct sysctl_oid *ctx_oid; 596 int i; 597 uint8_t name_str[16]; 598 599 ctx = device_get_sysctl_ctx(ha->pci_dev); 600 children = SYSCTL_CHILDREN(device_get_sysctl_tree(ha->pci_dev)); 601 602 ctx_oid = SYSCTL_ADD_NODE(ctx, children, OID_AUTO, "stats_hw_xmt", 603 CTLFLAG_RD | CTLFLAG_MPSAFE, NULL, "stats_hw_xmt"); 604 children = SYSCTL_CHILDREN(ctx_oid); 605 606 for (i = 0; i < ha->hw.num_tx_rings; i++) { 607 608 bzero(name_str, (sizeof(uint8_t) * sizeof(name_str))); 609 snprintf(name_str, sizeof(name_str), "%d", i); 610 611 ctx_oid = SYSCTL_ADD_NODE(ctx, children, OID_AUTO, name_str, 612 CTLFLAG_RD | CTLFLAG_MPSAFE, NULL, name_str); 613 node_children = SYSCTL_CHILDREN(ctx_oid); 614 615 /* Tx Related */ 616 617 SYSCTL_ADD_QUAD(ctx, node_children, 618 OID_AUTO, "total_bytes", 619 CTLFLAG_RD, &ha->hw.xmt[i].total_bytes, 620 "total_bytes"); 621 622 SYSCTL_ADD_QUAD(ctx, node_children, 623 OID_AUTO, "total_pkts", 624 CTLFLAG_RD, &ha->hw.xmt[i].total_pkts, 625 "total_pkts"); 626 627 SYSCTL_ADD_QUAD(ctx, node_children, 628 OID_AUTO, "errors", 629 CTLFLAG_RD, &ha->hw.xmt[i].errors, 630 "errors"); 631 632 SYSCTL_ADD_QUAD(ctx, node_children, 633 OID_AUTO, "pkts_dropped", 634 CTLFLAG_RD, &ha->hw.xmt[i].pkts_dropped, 635 "pkts_dropped"); 636 637 SYSCTL_ADD_QUAD(ctx, node_children, 638 OID_AUTO, "switch_pkts", 639 CTLFLAG_RD, &ha->hw.xmt[i].switch_pkts, 640 "switch_pkts"); 641 642 SYSCTL_ADD_QUAD(ctx, node_children, 643 OID_AUTO, "num_buffers", 644 CTLFLAG_RD, &ha->hw.xmt[i].num_buffers, 645 "num_buffers"); 646 } 647 648 return; 649 } 650 651 static void 652 qlnx_add_hw_mbx_cmpl_stats_sysctls(qla_host_t *ha) 653 { 654 struct sysctl_ctx_list *ctx; 655 struct sysctl_oid_list *node_children; 656 657 ctx = device_get_sysctl_ctx(ha->pci_dev); 658 node_children = SYSCTL_CHILDREN(device_get_sysctl_tree(ha->pci_dev)); 659 660 SYSCTL_ADD_QUAD(ctx, node_children, 661 OID_AUTO, "mbx_completion_time_lt_200ms", 662 CTLFLAG_RD, &ha->hw.mbx_comp_msecs[0], 663 "mbx_completion_time_lt_200ms"); 664 665 SYSCTL_ADD_QUAD(ctx, node_children, 666 OID_AUTO, "mbx_completion_time_200ms_400ms", 667 CTLFLAG_RD, &ha->hw.mbx_comp_msecs[1], 668 "mbx_completion_time_200ms_400ms"); 669 670 SYSCTL_ADD_QUAD(ctx, node_children, 671 OID_AUTO, "mbx_completion_time_400ms_600ms", 672 CTLFLAG_RD, &ha->hw.mbx_comp_msecs[2], 673 "mbx_completion_time_400ms_600ms"); 674 675 SYSCTL_ADD_QUAD(ctx, node_children, 676 OID_AUTO, "mbx_completion_time_600ms_800ms", 677 CTLFLAG_RD, &ha->hw.mbx_comp_msecs[3], 678 "mbx_completion_time_600ms_800ms"); 679 680 SYSCTL_ADD_QUAD(ctx, node_children, 681 OID_AUTO, "mbx_completion_time_800ms_1000ms", 682 CTLFLAG_RD, &ha->hw.mbx_comp_msecs[4], 683 "mbx_completion_time_800ms_1000ms"); 684 685 SYSCTL_ADD_QUAD(ctx, node_children, 686 OID_AUTO, "mbx_completion_time_1000ms_1200ms", 687 CTLFLAG_RD, &ha->hw.mbx_comp_msecs[5], 688 "mbx_completion_time_1000ms_1200ms"); 689 690 SYSCTL_ADD_QUAD(ctx, node_children, 691 OID_AUTO, "mbx_completion_time_1200ms_1400ms", 692 CTLFLAG_RD, &ha->hw.mbx_comp_msecs[6], 693 "mbx_completion_time_1200ms_1400ms"); 694 695 SYSCTL_ADD_QUAD(ctx, node_children, 696 OID_AUTO, "mbx_completion_time_1400ms_1600ms", 697 CTLFLAG_RD, &ha->hw.mbx_comp_msecs[7], 698 "mbx_completion_time_1400ms_1600ms"); 699 700 SYSCTL_ADD_QUAD(ctx, node_children, 701 OID_AUTO, "mbx_completion_time_1600ms_1800ms", 702 CTLFLAG_RD, &ha->hw.mbx_comp_msecs[8], 703 "mbx_completion_time_1600ms_1800ms"); 704 705 SYSCTL_ADD_QUAD(ctx, node_children, 706 OID_AUTO, "mbx_completion_time_1800ms_2000ms", 707 CTLFLAG_RD, &ha->hw.mbx_comp_msecs[9], 708 "mbx_completion_time_1800ms_2000ms"); 709 710 SYSCTL_ADD_QUAD(ctx, node_children, 711 OID_AUTO, "mbx_completion_time_2000ms_2200ms", 712 CTLFLAG_RD, &ha->hw.mbx_comp_msecs[10], 713 "mbx_completion_time_2000ms_2200ms"); 714 715 SYSCTL_ADD_QUAD(ctx, node_children, 716 OID_AUTO, "mbx_completion_time_2200ms_2400ms", 717 CTLFLAG_RD, &ha->hw.mbx_comp_msecs[11], 718 "mbx_completion_time_2200ms_2400ms"); 719 720 SYSCTL_ADD_QUAD(ctx, node_children, 721 OID_AUTO, "mbx_completion_time_2400ms_2600ms", 722 CTLFLAG_RD, &ha->hw.mbx_comp_msecs[12], 723 "mbx_completion_time_2400ms_2600ms"); 724 725 SYSCTL_ADD_QUAD(ctx, node_children, 726 OID_AUTO, "mbx_completion_time_2600ms_2800ms", 727 CTLFLAG_RD, &ha->hw.mbx_comp_msecs[13], 728 "mbx_completion_time_2600ms_2800ms"); 729 730 SYSCTL_ADD_QUAD(ctx, node_children, 731 OID_AUTO, "mbx_completion_time_2800ms_3000ms", 732 CTLFLAG_RD, &ha->hw.mbx_comp_msecs[14], 733 "mbx_completion_time_2800ms_3000ms"); 734 735 SYSCTL_ADD_QUAD(ctx, node_children, 736 OID_AUTO, "mbx_completion_time_3000ms_4000ms", 737 CTLFLAG_RD, &ha->hw.mbx_comp_msecs[15], 738 "mbx_completion_time_3000ms_4000ms"); 739 740 SYSCTL_ADD_QUAD(ctx, node_children, 741 OID_AUTO, "mbx_completion_time_4000ms_5000ms", 742 CTLFLAG_RD, &ha->hw.mbx_comp_msecs[16], 743 "mbx_completion_time_4000ms_5000ms"); 744 745 SYSCTL_ADD_QUAD(ctx, node_children, 746 OID_AUTO, "mbx_completion_host_mbx_cntrl_timeout", 747 CTLFLAG_RD, &ha->hw.mbx_comp_msecs[17], 748 "mbx_completion_host_mbx_cntrl_timeout"); 749 750 SYSCTL_ADD_QUAD(ctx, node_children, 751 OID_AUTO, "mbx_completion_fw_mbx_cntrl_timeout", 752 CTLFLAG_RD, &ha->hw.mbx_comp_msecs[18], 753 "mbx_completion_fw_mbx_cntrl_timeout"); 754 return; 755 } 756 757 static void 758 qlnx_add_hw_stats_sysctls(qla_host_t *ha) 759 { 760 qlnx_add_hw_mac_stats_sysctls(ha); 761 qlnx_add_hw_rcv_stats_sysctls(ha); 762 qlnx_add_hw_xmt_stats_sysctls(ha); 763 qlnx_add_hw_mbx_cmpl_stats_sysctls(ha); 764 765 return; 766 } 767 768 static void 769 qlnx_add_drvr_sds_stats(qla_host_t *ha) 770 { 771 struct sysctl_ctx_list *ctx; 772 struct sysctl_oid_list *children; 773 struct sysctl_oid_list *node_children; 774 struct sysctl_oid *ctx_oid; 775 int i; 776 uint8_t name_str[16]; 777 778 ctx = device_get_sysctl_ctx(ha->pci_dev); 779 children = SYSCTL_CHILDREN(device_get_sysctl_tree(ha->pci_dev)); 780 781 ctx_oid = SYSCTL_ADD_NODE(ctx, children, OID_AUTO, "stats_drvr_sds", 782 CTLFLAG_RD | CTLFLAG_MPSAFE, NULL, "stats_drvr_sds"); 783 children = SYSCTL_CHILDREN(ctx_oid); 784 785 for (i = 0; i < ha->hw.num_sds_rings; i++) { 786 787 bzero(name_str, (sizeof(uint8_t) * sizeof(name_str))); 788 snprintf(name_str, sizeof(name_str), "%d", i); 789 790 ctx_oid = SYSCTL_ADD_NODE(ctx, children, OID_AUTO, name_str, 791 CTLFLAG_RD | CTLFLAG_MPSAFE, NULL, name_str); 792 node_children = SYSCTL_CHILDREN(ctx_oid); 793 794 SYSCTL_ADD_QUAD(ctx, node_children, 795 OID_AUTO, "intr_count", 796 CTLFLAG_RD, &ha->hw.sds[i].intr_count, 797 "intr_count"); 798 799 SYSCTL_ADD_UINT(ctx, node_children, 800 OID_AUTO, "rx_free", 801 CTLFLAG_RD, &ha->hw.sds[i].rx_free, 802 ha->hw.sds[i].rx_free, "rx_free"); 803 } 804 805 return; 806 } 807 static void 808 qlnx_add_drvr_rds_stats(qla_host_t *ha) 809 { 810 struct sysctl_ctx_list *ctx; 811 struct sysctl_oid_list *children; 812 struct sysctl_oid_list *node_children; 813 struct sysctl_oid *ctx_oid; 814 int i; 815 uint8_t name_str[16]; 816 817 ctx = device_get_sysctl_ctx(ha->pci_dev); 818 children = SYSCTL_CHILDREN(device_get_sysctl_tree(ha->pci_dev)); 819 820 ctx_oid = SYSCTL_ADD_NODE(ctx, children, OID_AUTO, "stats_drvr_rds", 821 CTLFLAG_RD | CTLFLAG_MPSAFE, NULL, "stats_drvr_rds"); 822 children = SYSCTL_CHILDREN(ctx_oid); 823 824 for (i = 0; i < ha->hw.num_rds_rings; i++) { 825 826 bzero(name_str, (sizeof(uint8_t) * sizeof(name_str))); 827 snprintf(name_str, sizeof(name_str), "%d", i); 828 829 ctx_oid = SYSCTL_ADD_NODE(ctx, children, OID_AUTO, name_str, 830 CTLFLAG_RD | CTLFLAG_MPSAFE, NULL, name_str); 831 node_children = SYSCTL_CHILDREN(ctx_oid); 832 833 SYSCTL_ADD_QUAD(ctx, node_children, 834 OID_AUTO, "count", 835 CTLFLAG_RD, &ha->hw.rds[i].count, 836 "count"); 837 838 SYSCTL_ADD_QUAD(ctx, node_children, 839 OID_AUTO, "lro_pkt_count", 840 CTLFLAG_RD, &ha->hw.rds[i].lro_pkt_count, 841 "lro_pkt_count"); 842 843 SYSCTL_ADD_QUAD(ctx, node_children, 844 OID_AUTO, "lro_bytes", 845 CTLFLAG_RD, &ha->hw.rds[i].lro_bytes, 846 "lro_bytes"); 847 } 848 849 return; 850 } 851 852 static void 853 qlnx_add_drvr_tx_stats(qla_host_t *ha) 854 { 855 struct sysctl_ctx_list *ctx; 856 struct sysctl_oid_list *children; 857 struct sysctl_oid_list *node_children; 858 struct sysctl_oid *ctx_oid; 859 int i; 860 uint8_t name_str[16]; 861 862 ctx = device_get_sysctl_ctx(ha->pci_dev); 863 children = SYSCTL_CHILDREN(device_get_sysctl_tree(ha->pci_dev)); 864 865 ctx_oid = SYSCTL_ADD_NODE(ctx, children, OID_AUTO, "stats_drvr_xmt", 866 CTLFLAG_RD | CTLFLAG_MPSAFE, NULL, "stats_drvr_xmt"); 867 children = SYSCTL_CHILDREN(ctx_oid); 868 869 for (i = 0; i < ha->hw.num_tx_rings; i++) { 870 871 bzero(name_str, (sizeof(uint8_t) * sizeof(name_str))); 872 snprintf(name_str, sizeof(name_str), "%d", i); 873 874 ctx_oid = SYSCTL_ADD_NODE(ctx, children, OID_AUTO, name_str, 875 CTLFLAG_RD | CTLFLAG_MPSAFE, NULL, name_str); 876 node_children = SYSCTL_CHILDREN(ctx_oid); 877 878 SYSCTL_ADD_QUAD(ctx, node_children, 879 OID_AUTO, "count", 880 CTLFLAG_RD, &ha->tx_ring[i].count, 881 "count"); 882 883 #ifdef QL_ENABLE_ISCSI_TLV 884 SYSCTL_ADD_QUAD(ctx, node_children, 885 OID_AUTO, "iscsi_pkt_count", 886 CTLFLAG_RD, &ha->tx_ring[i].iscsi_pkt_count, 887 "iscsi_pkt_count"); 888 #endif /* #ifdef QL_ENABLE_ISCSI_TLV */ 889 } 890 891 return; 892 } 893 894 static void 895 qlnx_add_drvr_stats_sysctls(qla_host_t *ha) 896 { 897 qlnx_add_drvr_sds_stats(ha); 898 qlnx_add_drvr_rds_stats(ha); 899 qlnx_add_drvr_tx_stats(ha); 900 return; 901 } 902 903 /* 904 * Name: ql_hw_add_sysctls 905 * Function: Add P3Plus specific sysctls 906 */ 907 void 908 ql_hw_add_sysctls(qla_host_t *ha) 909 { 910 device_t dev; 911 912 dev = ha->pci_dev; 913 914 SYSCTL_ADD_UINT(device_get_sysctl_ctx(dev), 915 SYSCTL_CHILDREN(device_get_sysctl_tree(dev)), 916 OID_AUTO, "num_rds_rings", CTLFLAG_RD, &ha->hw.num_rds_rings, 917 ha->hw.num_rds_rings, "Number of Rcv Descriptor Rings"); 918 919 SYSCTL_ADD_UINT(device_get_sysctl_ctx(dev), 920 SYSCTL_CHILDREN(device_get_sysctl_tree(dev)), 921 OID_AUTO, "num_sds_rings", CTLFLAG_RD, &ha->hw.num_sds_rings, 922 ha->hw.num_sds_rings, "Number of Status Descriptor Rings"); 923 924 SYSCTL_ADD_UINT(device_get_sysctl_ctx(dev), 925 SYSCTL_CHILDREN(device_get_sysctl_tree(dev)), 926 OID_AUTO, "num_tx_rings", CTLFLAG_RD, &ha->hw.num_tx_rings, 927 ha->hw.num_tx_rings, "Number of Transmit Rings"); 928 929 SYSCTL_ADD_UINT(device_get_sysctl_ctx(dev), 930 SYSCTL_CHILDREN(device_get_sysctl_tree(dev)), 931 OID_AUTO, "tx_ring_index", CTLFLAG_RW, &ha->txr_idx, 932 ha->txr_idx, "Tx Ring Used"); 933 934 SYSCTL_ADD_UINT(device_get_sysctl_ctx(dev), 935 SYSCTL_CHILDREN(device_get_sysctl_tree(dev)), 936 OID_AUTO, "max_tx_segs", CTLFLAG_RD, &ha->hw.max_tx_segs, 937 ha->hw.max_tx_segs, "Max # of Segments in a non-TSO pkt"); 938 939 ha->hw.sds_cidx_thres = 32; 940 SYSCTL_ADD_UINT(device_get_sysctl_ctx(dev), 941 SYSCTL_CHILDREN(device_get_sysctl_tree(dev)), 942 OID_AUTO, "sds_cidx_thres", CTLFLAG_RW, &ha->hw.sds_cidx_thres, 943 ha->hw.sds_cidx_thres, 944 "Number of SDS entries to process before updating" 945 " SDS Ring Consumer Index"); 946 947 ha->hw.rds_pidx_thres = 32; 948 SYSCTL_ADD_UINT(device_get_sysctl_ctx(dev), 949 SYSCTL_CHILDREN(device_get_sysctl_tree(dev)), 950 OID_AUTO, "rds_pidx_thres", CTLFLAG_RW, &ha->hw.rds_pidx_thres, 951 ha->hw.rds_pidx_thres, 952 "Number of Rcv Rings Entries to post before updating" 953 " RDS Ring Producer Index"); 954 955 ha->hw.rcv_intr_coalesce = (3 << 16) | 256; 956 SYSCTL_ADD_UINT(device_get_sysctl_ctx(dev), 957 SYSCTL_CHILDREN(device_get_sysctl_tree(dev)), 958 OID_AUTO, "rcv_intr_coalesce", CTLFLAG_RW, 959 &ha->hw.rcv_intr_coalesce, 960 ha->hw.rcv_intr_coalesce, 961 "Rcv Intr Coalescing Parameters\n" 962 "\tbits 15:0 max packets\n" 963 "\tbits 31:16 max micro-seconds to wait\n" 964 "\tplease run\n" 965 "\tifconfig <if> down && ifconfig <if> up\n" 966 "\tto take effect \n"); 967 968 ha->hw.xmt_intr_coalesce = (64 << 16) | 64; 969 SYSCTL_ADD_UINT(device_get_sysctl_ctx(dev), 970 SYSCTL_CHILDREN(device_get_sysctl_tree(dev)), 971 OID_AUTO, "xmt_intr_coalesce", CTLFLAG_RW, 972 &ha->hw.xmt_intr_coalesce, 973 ha->hw.xmt_intr_coalesce, 974 "Xmt Intr Coalescing Parameters\n" 975 "\tbits 15:0 max packets\n" 976 "\tbits 31:16 max micro-seconds to wait\n" 977 "\tplease run\n" 978 "\tifconfig <if> down && ifconfig <if> up\n" 979 "\tto take effect \n"); 980 981 SYSCTL_ADD_PROC(device_get_sysctl_ctx(dev), 982 SYSCTL_CHILDREN(device_get_sysctl_tree(dev)), OID_AUTO, 983 "port_cfg", CTLTYPE_INT | CTLFLAG_RW | CTLFLAG_NEEDGIANT, 984 (void *)ha, 0, qla_sysctl_port_cfg, "I", 985 "Set Port Configuration if values below " 986 "otherwise Get Port Configuration\n" 987 "\tBits 0-3 ; 1 = DCBX Enable; 0 = DCBX Disable\n" 988 "\tBits 4-7 : 0 = no pause; 1 = std ; 2 = ppm \n" 989 "\tBits 8-11: std pause cfg; 0 = xmt and rcv;" 990 " 1 = xmt only; 2 = rcv only;\n"); 991 992 SYSCTL_ADD_PROC(device_get_sysctl_ctx(dev), 993 SYSCTL_CHILDREN(device_get_sysctl_tree(dev)), OID_AUTO, 994 "set_cam_search_mode", CTLTYPE_INT | CTLFLAG_RW | CTLFLAG_NEEDGIANT, 995 (void *)ha, 0, qla_sysctl_set_cam_search_mode, "I", 996 "Set CAM Search Mode" 997 "\t 1 = search mode internal\n" 998 "\t 2 = search mode auto\n"); 999 1000 SYSCTL_ADD_PROC(device_get_sysctl_ctx(dev), 1001 SYSCTL_CHILDREN(device_get_sysctl_tree(dev)), OID_AUTO, 1002 "get_cam_search_mode", 1003 CTLTYPE_INT | CTLFLAG_RW | CTLFLAG_NEEDGIANT, (void *)ha, 0, 1004 qla_sysctl_get_cam_search_mode, "I", 1005 "Get CAM Search Mode" 1006 "\t 1 = search mode internal\n" 1007 "\t 2 = search mode auto\n"); 1008 1009 ha->hw.enable_9kb = 1; 1010 1011 SYSCTL_ADD_UINT(device_get_sysctl_ctx(dev), 1012 SYSCTL_CHILDREN(device_get_sysctl_tree(dev)), 1013 OID_AUTO, "enable_9kb", CTLFLAG_RW, &ha->hw.enable_9kb, 1014 ha->hw.enable_9kb, "Enable 9Kbyte Buffers when MTU = 9000"); 1015 1016 ha->hw.enable_hw_lro = 1; 1017 1018 SYSCTL_ADD_UINT(device_get_sysctl_ctx(dev), 1019 SYSCTL_CHILDREN(device_get_sysctl_tree(dev)), 1020 OID_AUTO, "enable_hw_lro", CTLFLAG_RW, &ha->hw.enable_hw_lro, 1021 ha->hw.enable_hw_lro, "Enable Hardware LRO; Default is true \n" 1022 "\t 1 : Hardware LRO if LRO is enabled\n" 1023 "\t 0 : Software LRO if LRO is enabled\n" 1024 "\t Any change requires ifconfig down/up to take effect\n" 1025 "\t Note that LRO may be turned off/on via ifconfig\n"); 1026 1027 SYSCTL_ADD_UINT(device_get_sysctl_ctx(dev), 1028 SYSCTL_CHILDREN(device_get_sysctl_tree(dev)), 1029 OID_AUTO, "sp_log_index", CTLFLAG_RW, &ha->hw.sp_log_index, 1030 ha->hw.sp_log_index, "sp_log_index"); 1031 1032 SYSCTL_ADD_UINT(device_get_sysctl_ctx(dev), 1033 SYSCTL_CHILDREN(device_get_sysctl_tree(dev)), 1034 OID_AUTO, "sp_log_stop", CTLFLAG_RW, &ha->hw.sp_log_stop, 1035 ha->hw.sp_log_stop, "sp_log_stop"); 1036 1037 ha->hw.sp_log_stop_events = 0; 1038 1039 SYSCTL_ADD_UINT(device_get_sysctl_ctx(dev), 1040 SYSCTL_CHILDREN(device_get_sysctl_tree(dev)), 1041 OID_AUTO, "sp_log_stop_events", CTLFLAG_RW, 1042 &ha->hw.sp_log_stop_events, 1043 ha->hw.sp_log_stop_events, "Slow path event log is stopped" 1044 " when OR of the following events occur \n" 1045 "\t 0x01 : Heart beat Failure\n" 1046 "\t 0x02 : Temperature Failure\n" 1047 "\t 0x04 : HW Initialization Failure\n" 1048 "\t 0x08 : Interface Initialization Failure\n" 1049 "\t 0x10 : Error Recovery Failure\n"); 1050 1051 ha->hw.mdump_active = 0; 1052 SYSCTL_ADD_UINT(device_get_sysctl_ctx(dev), 1053 SYSCTL_CHILDREN(device_get_sysctl_tree(dev)), 1054 OID_AUTO, "minidump_active", CTLFLAG_RW, &ha->hw.mdump_active, 1055 ha->hw.mdump_active, 1056 "Minidump retrieval is Active"); 1057 1058 ha->hw.mdump_done = 0; 1059 SYSCTL_ADD_UINT(device_get_sysctl_ctx(dev), 1060 SYSCTL_CHILDREN(device_get_sysctl_tree(dev)), 1061 OID_AUTO, "mdump_done", CTLFLAG_RW, 1062 &ha->hw.mdump_done, ha->hw.mdump_done, 1063 "Minidump has been done and available for retrieval"); 1064 1065 ha->hw.mdump_capture_mask = 0xF; 1066 SYSCTL_ADD_UINT(device_get_sysctl_ctx(dev), 1067 SYSCTL_CHILDREN(device_get_sysctl_tree(dev)), 1068 OID_AUTO, "minidump_capture_mask", CTLFLAG_RW, 1069 &ha->hw.mdump_capture_mask, ha->hw.mdump_capture_mask, 1070 "Minidump capture mask"); 1071 #ifdef QL_DBG 1072 1073 ha->err_inject = 0; 1074 SYSCTL_ADD_UINT(device_get_sysctl_ctx(dev), 1075 SYSCTL_CHILDREN(device_get_sysctl_tree(dev)), 1076 OID_AUTO, "err_inject", 1077 CTLFLAG_RW, &ha->err_inject, ha->err_inject, 1078 "Error to be injected\n" 1079 "\t\t\t 0: No Errors\n" 1080 "\t\t\t 1: rcv: rxb struct invalid\n" 1081 "\t\t\t 2: rcv: mp == NULL\n" 1082 "\t\t\t 3: lro: rxb struct invalid\n" 1083 "\t\t\t 4: lro: mp == NULL\n" 1084 "\t\t\t 5: rcv: num handles invalid\n" 1085 "\t\t\t 6: reg: indirect reg rd_wr failure\n" 1086 "\t\t\t 7: ocm: offchip memory rd_wr failure\n" 1087 "\t\t\t 8: mbx: mailbox command failure\n" 1088 "\t\t\t 9: heartbeat failure\n" 1089 "\t\t\t A: temperature failure\n" 1090 "\t\t\t 11: m_getcl or m_getjcl failure\n" 1091 "\t\t\t 13: Invalid Descriptor Count in SGL Receive\n" 1092 "\t\t\t 14: Invalid Descriptor Count in LRO Receive\n" 1093 "\t\t\t 15: peer port error recovery failure\n" 1094 "\t\t\t 16: tx_buf[next_prod_index].mbuf != NULL\n" ); 1095 1096 SYSCTL_ADD_PROC(device_get_sysctl_ctx(dev), 1097 SYSCTL_CHILDREN(device_get_sysctl_tree(dev)), OID_AUTO, 1098 "peg_stop", CTLTYPE_INT | CTLFLAG_RW | CTLFLAG_NEEDGIANT, 1099 (void *)ha, 0, qla_sysctl_stop_pegs, "I", "Peg Stop"); 1100 1101 #endif /* #ifdef QL_DBG */ 1102 1103 ha->hw.user_pri_nic = 0; 1104 SYSCTL_ADD_UINT(device_get_sysctl_ctx(dev), 1105 SYSCTL_CHILDREN(device_get_sysctl_tree(dev)), 1106 OID_AUTO, "user_pri_nic", CTLFLAG_RW, &ha->hw.user_pri_nic, 1107 ha->hw.user_pri_nic, 1108 "VLAN Tag User Priority for Normal Ethernet Packets"); 1109 1110 ha->hw.user_pri_iscsi = 4; 1111 SYSCTL_ADD_UINT(device_get_sysctl_ctx(dev), 1112 SYSCTL_CHILDREN(device_get_sysctl_tree(dev)), 1113 OID_AUTO, "user_pri_iscsi", CTLFLAG_RW, &ha->hw.user_pri_iscsi, 1114 ha->hw.user_pri_iscsi, 1115 "VLAN Tag User Priority for iSCSI Packets"); 1116 1117 qlnx_add_hw_stats_sysctls(ha); 1118 qlnx_add_drvr_stats_sysctls(ha); 1119 1120 return; 1121 } 1122 1123 void 1124 ql_hw_link_status(qla_host_t *ha) 1125 { 1126 device_printf(ha->pci_dev, "cable_oui\t\t 0x%08x\n", ha->hw.cable_oui); 1127 1128 if (ha->hw.link_up) { 1129 device_printf(ha->pci_dev, "link Up\n"); 1130 } else { 1131 device_printf(ha->pci_dev, "link Down\n"); 1132 } 1133 1134 if (ha->hw.fduplex) { 1135 device_printf(ha->pci_dev, "Full Duplex\n"); 1136 } else { 1137 device_printf(ha->pci_dev, "Half Duplex\n"); 1138 } 1139 1140 if (ha->hw.autoneg) { 1141 device_printf(ha->pci_dev, "Auto Negotiation Enabled\n"); 1142 } else { 1143 device_printf(ha->pci_dev, "Auto Negotiation Disabled\n"); 1144 } 1145 1146 switch (ha->hw.link_speed) { 1147 case 0x710: 1148 device_printf(ha->pci_dev, "link speed\t\t 10Gps\n"); 1149 break; 1150 1151 case 0x3E8: 1152 device_printf(ha->pci_dev, "link speed\t\t 1Gps\n"); 1153 break; 1154 1155 case 0x64: 1156 device_printf(ha->pci_dev, "link speed\t\t 100Mbps\n"); 1157 break; 1158 1159 default: 1160 device_printf(ha->pci_dev, "link speed\t\t Unknown\n"); 1161 break; 1162 } 1163 1164 switch (ha->hw.module_type) { 1165 1166 case 0x01: 1167 device_printf(ha->pci_dev, "Module Type 10GBase-LRM\n"); 1168 break; 1169 1170 case 0x02: 1171 device_printf(ha->pci_dev, "Module Type 10GBase-LR\n"); 1172 break; 1173 1174 case 0x03: 1175 device_printf(ha->pci_dev, "Module Type 10GBase-SR\n"); 1176 break; 1177 1178 case 0x04: 1179 device_printf(ha->pci_dev, 1180 "Module Type 10GE Passive Copper(Compliant)[%d m]\n", 1181 ha->hw.cable_length); 1182 break; 1183 1184 case 0x05: 1185 device_printf(ha->pci_dev, "Module Type 10GE Active" 1186 " Limiting Copper(Compliant)[%d m]\n", 1187 ha->hw.cable_length); 1188 break; 1189 1190 case 0x06: 1191 device_printf(ha->pci_dev, 1192 "Module Type 10GE Passive Copper" 1193 " (Legacy, Best Effort)[%d m]\n", 1194 ha->hw.cable_length); 1195 break; 1196 1197 case 0x07: 1198 device_printf(ha->pci_dev, "Module Type 1000Base-SX\n"); 1199 break; 1200 1201 case 0x08: 1202 device_printf(ha->pci_dev, "Module Type 1000Base-LX\n"); 1203 break; 1204 1205 case 0x09: 1206 device_printf(ha->pci_dev, "Module Type 1000Base-CX\n"); 1207 break; 1208 1209 case 0x0A: 1210 device_printf(ha->pci_dev, "Module Type 1000Base-T\n"); 1211 break; 1212 1213 case 0x0B: 1214 device_printf(ha->pci_dev, "Module Type 1GE Passive Copper" 1215 "(Legacy, Best Effort)\n"); 1216 break; 1217 1218 default: 1219 device_printf(ha->pci_dev, "Unknown Module Type 0x%x\n", 1220 ha->hw.module_type); 1221 break; 1222 } 1223 1224 if (ha->hw.link_faults == 1) 1225 device_printf(ha->pci_dev, "SFP Power Fault\n"); 1226 } 1227 1228 /* 1229 * Name: ql_free_dma 1230 * Function: Frees the DMA'able memory allocated in ql_alloc_dma() 1231 */ 1232 void 1233 ql_free_dma(qla_host_t *ha) 1234 { 1235 uint32_t i; 1236 1237 if (ha->hw.dma_buf.flags.sds_ring) { 1238 for (i = 0; i < ha->hw.num_sds_rings; i++) { 1239 ql_free_dmabuf(ha, &ha->hw.dma_buf.sds_ring[i]); 1240 } 1241 ha->hw.dma_buf.flags.sds_ring = 0; 1242 } 1243 1244 if (ha->hw.dma_buf.flags.rds_ring) { 1245 for (i = 0; i < ha->hw.num_rds_rings; i++) { 1246 ql_free_dmabuf(ha, &ha->hw.dma_buf.rds_ring[i]); 1247 } 1248 ha->hw.dma_buf.flags.rds_ring = 0; 1249 } 1250 1251 if (ha->hw.dma_buf.flags.tx_ring) { 1252 ql_free_dmabuf(ha, &ha->hw.dma_buf.tx_ring); 1253 ha->hw.dma_buf.flags.tx_ring = 0; 1254 } 1255 ql_minidump_free(ha); 1256 } 1257 1258 /* 1259 * Name: ql_alloc_dma 1260 * Function: Allocates DMA'able memory for Tx/Rx Rings, Tx/Rx Contexts. 1261 */ 1262 int 1263 ql_alloc_dma(qla_host_t *ha) 1264 { 1265 device_t dev; 1266 uint32_t i, j, size, tx_ring_size; 1267 qla_hw_t *hw; 1268 qla_hw_tx_cntxt_t *tx_cntxt; 1269 uint8_t *vaddr; 1270 bus_addr_t paddr; 1271 1272 dev = ha->pci_dev; 1273 1274 QL_DPRINT2(ha, (dev, "%s: enter\n", __func__)); 1275 1276 hw = &ha->hw; 1277 /* 1278 * Allocate Transmit Ring 1279 */ 1280 tx_ring_size = (sizeof(q80_tx_cmd_t) * NUM_TX_DESCRIPTORS); 1281 size = (tx_ring_size * ha->hw.num_tx_rings); 1282 1283 hw->dma_buf.tx_ring.alignment = 8; 1284 hw->dma_buf.tx_ring.size = size + PAGE_SIZE; 1285 1286 if (ql_alloc_dmabuf(ha, &hw->dma_buf.tx_ring)) { 1287 device_printf(dev, "%s: tx ring alloc failed\n", __func__); 1288 goto ql_alloc_dma_exit; 1289 } 1290 1291 vaddr = (uint8_t *)hw->dma_buf.tx_ring.dma_b; 1292 paddr = hw->dma_buf.tx_ring.dma_addr; 1293 1294 for (i = 0; i < ha->hw.num_tx_rings; i++) { 1295 tx_cntxt = (qla_hw_tx_cntxt_t *)&hw->tx_cntxt[i]; 1296 1297 tx_cntxt->tx_ring_base = (q80_tx_cmd_t *)vaddr; 1298 tx_cntxt->tx_ring_paddr = paddr; 1299 1300 vaddr += tx_ring_size; 1301 paddr += tx_ring_size; 1302 } 1303 1304 for (i = 0; i < ha->hw.num_tx_rings; i++) { 1305 tx_cntxt = (qla_hw_tx_cntxt_t *)&hw->tx_cntxt[i]; 1306 1307 tx_cntxt->tx_cons = (uint32_t *)vaddr; 1308 tx_cntxt->tx_cons_paddr = paddr; 1309 1310 vaddr += sizeof (uint32_t); 1311 paddr += sizeof (uint32_t); 1312 } 1313 1314 ha->hw.dma_buf.flags.tx_ring = 1; 1315 1316 QL_DPRINT2(ha, (dev, "%s: tx_ring phys %p virt %p\n", 1317 __func__, (void *)(hw->dma_buf.tx_ring.dma_addr), 1318 hw->dma_buf.tx_ring.dma_b)); 1319 /* 1320 * Allocate Receive Descriptor Rings 1321 */ 1322 1323 for (i = 0; i < hw->num_rds_rings; i++) { 1324 1325 hw->dma_buf.rds_ring[i].alignment = 8; 1326 hw->dma_buf.rds_ring[i].size = 1327 (sizeof(q80_recv_desc_t)) * NUM_RX_DESCRIPTORS; 1328 1329 if (ql_alloc_dmabuf(ha, &hw->dma_buf.rds_ring[i])) { 1330 device_printf(dev, "%s: rds ring[%d] alloc failed\n", 1331 __func__, i); 1332 1333 for (j = 0; j < i; j++) 1334 ql_free_dmabuf(ha, &hw->dma_buf.rds_ring[j]); 1335 1336 goto ql_alloc_dma_exit; 1337 } 1338 QL_DPRINT4(ha, (dev, "%s: rx_ring[%d] phys %p virt %p\n", 1339 __func__, i, (void *)(hw->dma_buf.rds_ring[i].dma_addr), 1340 hw->dma_buf.rds_ring[i].dma_b)); 1341 } 1342 1343 hw->dma_buf.flags.rds_ring = 1; 1344 1345 /* 1346 * Allocate Status Descriptor Rings 1347 */ 1348 1349 for (i = 0; i < hw->num_sds_rings; i++) { 1350 hw->dma_buf.sds_ring[i].alignment = 8; 1351 hw->dma_buf.sds_ring[i].size = 1352 (sizeof(q80_stat_desc_t)) * NUM_STATUS_DESCRIPTORS; 1353 1354 if (ql_alloc_dmabuf(ha, &hw->dma_buf.sds_ring[i])) { 1355 device_printf(dev, "%s: sds ring alloc failed\n", 1356 __func__); 1357 1358 for (j = 0; j < i; j++) 1359 ql_free_dmabuf(ha, &hw->dma_buf.sds_ring[j]); 1360 1361 goto ql_alloc_dma_exit; 1362 } 1363 QL_DPRINT4(ha, (dev, "%s: sds_ring[%d] phys %p virt %p\n", 1364 __func__, i, 1365 (void *)(hw->dma_buf.sds_ring[i].dma_addr), 1366 hw->dma_buf.sds_ring[i].dma_b)); 1367 } 1368 for (i = 0; i < hw->num_sds_rings; i++) { 1369 hw->sds[i].sds_ring_base = 1370 (q80_stat_desc_t *)hw->dma_buf.sds_ring[i].dma_b; 1371 } 1372 1373 hw->dma_buf.flags.sds_ring = 1; 1374 1375 return 0; 1376 1377 ql_alloc_dma_exit: 1378 ql_free_dma(ha); 1379 return -1; 1380 } 1381 1382 #define Q8_MBX_MSEC_DELAY 5000 1383 1384 static int 1385 qla_mbx_cmd(qla_host_t *ha, uint32_t *h_mbox, uint32_t n_hmbox, 1386 uint32_t *fw_mbox, uint32_t n_fwmbox, uint32_t no_pause) 1387 { 1388 uint32_t i; 1389 uint32_t data; 1390 int ret = 0; 1391 uint64_t start_usecs; 1392 uint64_t end_usecs; 1393 uint64_t msecs_200; 1394 1395 ql_sp_log(ha, 0, 5, no_pause, h_mbox[0], h_mbox[1], h_mbox[2], h_mbox[3]); 1396 1397 if (ha->offline || ha->qla_initiate_recovery) { 1398 ql_sp_log(ha, 1, 2, ha->offline, ha->qla_initiate_recovery, 0, 0, 0); 1399 goto exit_qla_mbx_cmd; 1400 } 1401 1402 if (((ha->err_inject & 0xFFFF) == INJCT_MBX_CMD_FAILURE) && 1403 (((ha->err_inject & ~0xFFFF) == ((h_mbox[0] & 0xFFFF) << 16))|| 1404 !(ha->err_inject & ~0xFFFF))) { 1405 ret = -3; 1406 QL_INITIATE_RECOVERY(ha); 1407 goto exit_qla_mbx_cmd; 1408 } 1409 1410 start_usecs = qla_get_usec_timestamp(); 1411 1412 if (no_pause) 1413 i = 1000; 1414 else 1415 i = Q8_MBX_MSEC_DELAY; 1416 1417 while (i) { 1418 1419 if (ha->qla_initiate_recovery) { 1420 ql_sp_log(ha, 2, 1, ha->qla_initiate_recovery, 0, 0, 0, 0); 1421 return (-1); 1422 } 1423 1424 data = READ_REG32(ha, Q8_HOST_MBOX_CNTRL); 1425 if (data == 0) 1426 break; 1427 if (no_pause) { 1428 DELAY(1000); 1429 } else { 1430 qla_mdelay(__func__, 1); 1431 } 1432 i--; 1433 } 1434 1435 if (i == 0) { 1436 device_printf(ha->pci_dev, "%s: host_mbx_cntrl 0x%08x\n", 1437 __func__, data); 1438 ql_sp_log(ha, 3, 1, data, 0, 0, 0, 0); 1439 ret = -1; 1440 ha->hw.mbx_comp_msecs[(Q8_MBX_COMP_MSECS - 2)]++; 1441 QL_INITIATE_RECOVERY(ha); 1442 goto exit_qla_mbx_cmd; 1443 } 1444 1445 for (i = 0; i < n_hmbox; i++) { 1446 WRITE_REG32(ha, (Q8_HOST_MBOX0 + (i << 2)), *h_mbox); 1447 h_mbox++; 1448 } 1449 1450 WRITE_REG32(ha, Q8_HOST_MBOX_CNTRL, 0x1); 1451 1452 1453 i = Q8_MBX_MSEC_DELAY; 1454 while (i) { 1455 1456 if (ha->qla_initiate_recovery) { 1457 ql_sp_log(ha, 4, 1, ha->qla_initiate_recovery, 0, 0, 0, 0); 1458 return (-1); 1459 } 1460 1461 data = READ_REG32(ha, Q8_FW_MBOX_CNTRL); 1462 1463 if ((data & 0x3) == 1) { 1464 data = READ_REG32(ha, Q8_FW_MBOX0); 1465 if ((data & 0xF000) != 0x8000) 1466 break; 1467 } 1468 if (no_pause) { 1469 DELAY(1000); 1470 } else { 1471 qla_mdelay(__func__, 1); 1472 } 1473 i--; 1474 } 1475 if (i == 0) { 1476 device_printf(ha->pci_dev, "%s: fw_mbx_cntrl 0x%08x\n", 1477 __func__, data); 1478 ql_sp_log(ha, 5, 1, data, 0, 0, 0, 0); 1479 ret = -2; 1480 ha->hw.mbx_comp_msecs[(Q8_MBX_COMP_MSECS - 1)]++; 1481 QL_INITIATE_RECOVERY(ha); 1482 goto exit_qla_mbx_cmd; 1483 } 1484 1485 for (i = 0; i < n_fwmbox; i++) { 1486 1487 if (ha->qla_initiate_recovery) { 1488 ql_sp_log(ha, 6, 1, ha->qla_initiate_recovery, 0, 0, 0, 0); 1489 return (-1); 1490 } 1491 1492 *fw_mbox++ = READ_REG32(ha, (Q8_FW_MBOX0 + (i << 2))); 1493 } 1494 1495 WRITE_REG32(ha, Q8_FW_MBOX_CNTRL, 0x0); 1496 WRITE_REG32(ha, ha->hw.mbx_intr_mask_offset, 0x0); 1497 1498 end_usecs = qla_get_usec_timestamp(); 1499 1500 if (end_usecs > start_usecs) { 1501 msecs_200 = (end_usecs - start_usecs)/(1000 * 200); 1502 1503 if (msecs_200 < 15) 1504 ha->hw.mbx_comp_msecs[msecs_200]++; 1505 else if (msecs_200 < 20) 1506 ha->hw.mbx_comp_msecs[15]++; 1507 else { 1508 device_printf(ha->pci_dev, "%s: [%ld, %ld] %ld\n", __func__, 1509 start_usecs, end_usecs, msecs_200); 1510 ha->hw.mbx_comp_msecs[16]++; 1511 } 1512 } 1513 ql_sp_log(ha, 7, 5, fw_mbox[0], fw_mbox[1], fw_mbox[2], fw_mbox[3], fw_mbox[4]); 1514 1515 1516 exit_qla_mbx_cmd: 1517 return (ret); 1518 } 1519 1520 int 1521 qla_get_nic_partition(qla_host_t *ha, uint32_t *supports_9kb, 1522 uint32_t *num_rcvq) 1523 { 1524 uint32_t *mbox, err; 1525 device_t dev = ha->pci_dev; 1526 1527 bzero(ha->hw.mbox, (sizeof (uint32_t) * Q8_NUM_MBOX)); 1528 1529 mbox = ha->hw.mbox; 1530 1531 mbox[0] = Q8_MBX_GET_NIC_PARTITION | (0x2 << 16) | (0x2 << 29); 1532 1533 if (qla_mbx_cmd(ha, mbox, 2, mbox, 19, 0)) { 1534 device_printf(dev, "%s: failed0\n", __func__); 1535 return (-1); 1536 } 1537 err = mbox[0] >> 25; 1538 1539 if (supports_9kb != NULL) { 1540 if (mbox[16] & 0x80) /* bit 7 of mbox 16 */ 1541 *supports_9kb = 1; 1542 else 1543 *supports_9kb = 0; 1544 } 1545 1546 if (num_rcvq != NULL) 1547 *num_rcvq = ((mbox[6] >> 16) & 0xFFFF); 1548 1549 if ((err != 1) && (err != 0)) { 1550 device_printf(dev, "%s: failed1 [0x%08x]\n", __func__, err); 1551 return (-1); 1552 } 1553 return 0; 1554 } 1555 1556 static int 1557 qla_config_intr_cntxt(qla_host_t *ha, uint32_t start_idx, uint32_t num_intrs, 1558 uint32_t create) 1559 { 1560 uint32_t i, err; 1561 device_t dev = ha->pci_dev; 1562 q80_config_intr_t *c_intr; 1563 q80_config_intr_rsp_t *c_intr_rsp; 1564 1565 c_intr = (q80_config_intr_t *)ha->hw.mbox; 1566 bzero(c_intr, (sizeof (q80_config_intr_t))); 1567 1568 c_intr->opcode = Q8_MBX_CONFIG_INTR; 1569 1570 c_intr->count_version = (sizeof (q80_config_intr_t) >> 2); 1571 c_intr->count_version |= Q8_MBX_CMD_VERSION; 1572 1573 c_intr->nentries = num_intrs; 1574 1575 for (i = 0; i < num_intrs; i++) { 1576 if (create) { 1577 c_intr->intr[i].cmd_type = Q8_MBX_CONFIG_INTR_CREATE; 1578 c_intr->intr[i].msix_index = start_idx + 1 + i; 1579 } else { 1580 c_intr->intr[i].cmd_type = Q8_MBX_CONFIG_INTR_DELETE; 1581 c_intr->intr[i].msix_index = 1582 ha->hw.intr_id[(start_idx + i)]; 1583 } 1584 1585 c_intr->intr[i].cmd_type |= Q8_MBX_CONFIG_INTR_TYPE_MSI_X; 1586 } 1587 1588 if (qla_mbx_cmd(ha, (uint32_t *)c_intr, 1589 (sizeof (q80_config_intr_t) >> 2), 1590 ha->hw.mbox, (sizeof (q80_config_intr_rsp_t) >> 2), 0)) { 1591 device_printf(dev, "%s: %s failed0\n", __func__, 1592 (create ? "create" : "delete")); 1593 return (-1); 1594 } 1595 1596 c_intr_rsp = (q80_config_intr_rsp_t *)ha->hw.mbox; 1597 1598 err = Q8_MBX_RSP_STATUS(c_intr_rsp->regcnt_status); 1599 1600 if (err) { 1601 device_printf(dev, "%s: %s failed1 [0x%08x, %d]\n", __func__, 1602 (create ? "create" : "delete"), err, c_intr_rsp->nentries); 1603 1604 for (i = 0; i < c_intr_rsp->nentries; i++) { 1605 device_printf(dev, "%s: [%d]:[0x%x 0x%x 0x%x]\n", 1606 __func__, i, 1607 c_intr_rsp->intr[i].status, 1608 c_intr_rsp->intr[i].intr_id, 1609 c_intr_rsp->intr[i].intr_src); 1610 } 1611 1612 return (-1); 1613 } 1614 1615 for (i = 0; ((i < num_intrs) && create); i++) { 1616 if (!c_intr_rsp->intr[i].status) { 1617 ha->hw.intr_id[(start_idx + i)] = 1618 c_intr_rsp->intr[i].intr_id; 1619 ha->hw.intr_src[(start_idx + i)] = 1620 c_intr_rsp->intr[i].intr_src; 1621 } 1622 } 1623 1624 return (0); 1625 } 1626 1627 /* 1628 * Name: qla_config_rss 1629 * Function: Configure RSS for the context/interface. 1630 */ 1631 static const uint64_t rss_key[] = { 0xbeac01fa6a42b73bULL, 1632 0x8030f20c77cb2da3ULL, 1633 0xae7b30b4d0ca2bcbULL, 0x43a38fb04167253dULL, 1634 0x255b0ec26d5a56daULL }; 1635 1636 static int 1637 qla_config_rss(qla_host_t *ha, uint16_t cntxt_id) 1638 { 1639 q80_config_rss_t *c_rss; 1640 q80_config_rss_rsp_t *c_rss_rsp; 1641 uint32_t err, i; 1642 device_t dev = ha->pci_dev; 1643 1644 c_rss = (q80_config_rss_t *)ha->hw.mbox; 1645 bzero(c_rss, (sizeof (q80_config_rss_t))); 1646 1647 c_rss->opcode = Q8_MBX_CONFIG_RSS; 1648 1649 c_rss->count_version = (sizeof (q80_config_rss_t) >> 2); 1650 c_rss->count_version |= Q8_MBX_CMD_VERSION; 1651 1652 c_rss->hash_type = (Q8_MBX_RSS_HASH_TYPE_IPV4_TCP_IP | 1653 Q8_MBX_RSS_HASH_TYPE_IPV6_TCP_IP); 1654 //c_rss->hash_type = (Q8_MBX_RSS_HASH_TYPE_IPV4_TCP | 1655 // Q8_MBX_RSS_HASH_TYPE_IPV6_TCP); 1656 1657 c_rss->flags = Q8_MBX_RSS_FLAGS_ENABLE_RSS; 1658 c_rss->flags |= Q8_MBX_RSS_FLAGS_USE_IND_TABLE; 1659 1660 c_rss->indtbl_mask = Q8_MBX_RSS_INDTBL_MASK; 1661 1662 c_rss->indtbl_mask |= Q8_MBX_RSS_FLAGS_MULTI_RSS_VALID; 1663 c_rss->flags |= Q8_MBX_RSS_FLAGS_TYPE_CRSS; 1664 1665 c_rss->cntxt_id = cntxt_id; 1666 1667 for (i = 0; i < 5; i++) { 1668 c_rss->rss_key[i] = rss_key[i]; 1669 } 1670 1671 if (qla_mbx_cmd(ha, (uint32_t *)c_rss, 1672 (sizeof (q80_config_rss_t) >> 2), 1673 ha->hw.mbox, (sizeof(q80_config_rss_rsp_t) >> 2), 0)) { 1674 device_printf(dev, "%s: failed0\n", __func__); 1675 return (-1); 1676 } 1677 c_rss_rsp = (q80_config_rss_rsp_t *)ha->hw.mbox; 1678 1679 err = Q8_MBX_RSP_STATUS(c_rss_rsp->regcnt_status); 1680 1681 if (err) { 1682 device_printf(dev, "%s: failed1 [0x%08x]\n", __func__, err); 1683 return (-1); 1684 } 1685 return 0; 1686 } 1687 1688 static int 1689 qla_set_rss_ind_table(qla_host_t *ha, uint32_t start_idx, uint32_t count, 1690 uint16_t cntxt_id, uint8_t *ind_table) 1691 { 1692 q80_config_rss_ind_table_t *c_rss_ind; 1693 q80_config_rss_ind_table_rsp_t *c_rss_ind_rsp; 1694 uint32_t err; 1695 device_t dev = ha->pci_dev; 1696 1697 if ((count > Q8_RSS_IND_TBL_SIZE) || 1698 ((start_idx + count - 1) > Q8_RSS_IND_TBL_MAX_IDX)) { 1699 device_printf(dev, "%s: illegal count [%d, %d]\n", __func__, 1700 start_idx, count); 1701 return (-1); 1702 } 1703 1704 c_rss_ind = (q80_config_rss_ind_table_t *)ha->hw.mbox; 1705 bzero(c_rss_ind, sizeof (q80_config_rss_ind_table_t)); 1706 1707 c_rss_ind->opcode = Q8_MBX_CONFIG_RSS_TABLE; 1708 c_rss_ind->count_version = (sizeof (q80_config_rss_ind_table_t) >> 2); 1709 c_rss_ind->count_version |= Q8_MBX_CMD_VERSION; 1710 1711 c_rss_ind->start_idx = start_idx; 1712 c_rss_ind->end_idx = start_idx + count - 1; 1713 c_rss_ind->cntxt_id = cntxt_id; 1714 bcopy(ind_table, c_rss_ind->ind_table, count); 1715 1716 if (qla_mbx_cmd(ha, (uint32_t *)c_rss_ind, 1717 (sizeof (q80_config_rss_ind_table_t) >> 2), ha->hw.mbox, 1718 (sizeof(q80_config_rss_ind_table_rsp_t) >> 2), 0)) { 1719 device_printf(dev, "%s: failed0\n", __func__); 1720 return (-1); 1721 } 1722 1723 c_rss_ind_rsp = (q80_config_rss_ind_table_rsp_t *)ha->hw.mbox; 1724 err = Q8_MBX_RSP_STATUS(c_rss_ind_rsp->regcnt_status); 1725 1726 if (err) { 1727 device_printf(dev, "%s: failed1 [0x%08x]\n", __func__, err); 1728 return (-1); 1729 } 1730 return 0; 1731 } 1732 1733 /* 1734 * Name: qla_config_intr_coalesce 1735 * Function: Configure Interrupt Coalescing. 1736 */ 1737 static int 1738 qla_config_intr_coalesce(qla_host_t *ha, uint16_t cntxt_id, int tenable, 1739 int rcv) 1740 { 1741 q80_config_intr_coalesc_t *intrc; 1742 q80_config_intr_coalesc_rsp_t *intrc_rsp; 1743 uint32_t err, i; 1744 device_t dev = ha->pci_dev; 1745 1746 intrc = (q80_config_intr_coalesc_t *)ha->hw.mbox; 1747 bzero(intrc, (sizeof (q80_config_intr_coalesc_t))); 1748 1749 intrc->opcode = Q8_MBX_CONFIG_INTR_COALESCE; 1750 intrc->count_version = (sizeof (q80_config_intr_coalesc_t) >> 2); 1751 intrc->count_version |= Q8_MBX_CMD_VERSION; 1752 1753 if (rcv) { 1754 intrc->flags = Q8_MBX_INTRC_FLAGS_RCV; 1755 intrc->max_pkts = ha->hw.rcv_intr_coalesce & 0xFFFF; 1756 intrc->max_mswait = (ha->hw.rcv_intr_coalesce >> 16) & 0xFFFF; 1757 } else { 1758 intrc->flags = Q8_MBX_INTRC_FLAGS_XMT; 1759 intrc->max_pkts = ha->hw.xmt_intr_coalesce & 0xFFFF; 1760 intrc->max_mswait = (ha->hw.xmt_intr_coalesce >> 16) & 0xFFFF; 1761 } 1762 1763 intrc->cntxt_id = cntxt_id; 1764 1765 if (tenable) { 1766 intrc->flags |= Q8_MBX_INTRC_FLAGS_PERIODIC; 1767 intrc->timer_type = Q8_MBX_INTRC_TIMER_PERIODIC; 1768 1769 for (i = 0; i < ha->hw.num_sds_rings; i++) { 1770 intrc->sds_ring_mask |= (1 << i); 1771 } 1772 intrc->ms_timeout = 1000; 1773 } 1774 1775 if (qla_mbx_cmd(ha, (uint32_t *)intrc, 1776 (sizeof (q80_config_intr_coalesc_t) >> 2), 1777 ha->hw.mbox, (sizeof(q80_config_intr_coalesc_rsp_t) >> 2), 0)) { 1778 device_printf(dev, "%s: failed0\n", __func__); 1779 return (-1); 1780 } 1781 intrc_rsp = (q80_config_intr_coalesc_rsp_t *)ha->hw.mbox; 1782 1783 err = Q8_MBX_RSP_STATUS(intrc_rsp->regcnt_status); 1784 1785 if (err) { 1786 device_printf(dev, "%s: failed1 [0x%08x]\n", __func__, err); 1787 return (-1); 1788 } 1789 1790 return 0; 1791 } 1792 1793 1794 /* 1795 * Name: qla_config_mac_addr 1796 * Function: binds a MAC address to the context/interface. 1797 * Can be unicast, multicast or broadcast. 1798 */ 1799 static int 1800 qla_config_mac_addr(qla_host_t *ha, uint8_t *mac_addr, uint32_t add_mac, 1801 uint32_t num_mac) 1802 { 1803 q80_config_mac_addr_t *cmac; 1804 q80_config_mac_addr_rsp_t *cmac_rsp; 1805 uint32_t err; 1806 device_t dev = ha->pci_dev; 1807 int i; 1808 uint8_t *mac_cpy = mac_addr; 1809 1810 if (num_mac > Q8_MAX_MAC_ADDRS) { 1811 device_printf(dev, "%s: %s num_mac [0x%x] > Q8_MAX_MAC_ADDRS\n", 1812 __func__, (add_mac ? "Add" : "Del"), num_mac); 1813 return (-1); 1814 } 1815 1816 cmac = (q80_config_mac_addr_t *)ha->hw.mbox; 1817 bzero(cmac, (sizeof (q80_config_mac_addr_t))); 1818 1819 cmac->opcode = Q8_MBX_CONFIG_MAC_ADDR; 1820 cmac->count_version = sizeof (q80_config_mac_addr_t) >> 2; 1821 cmac->count_version |= Q8_MBX_CMD_VERSION; 1822 1823 if (add_mac) 1824 cmac->cmd = Q8_MBX_CMAC_CMD_ADD_MAC_ADDR; 1825 else 1826 cmac->cmd = Q8_MBX_CMAC_CMD_DEL_MAC_ADDR; 1827 1828 cmac->cmd |= Q8_MBX_CMAC_CMD_CAM_INGRESS; 1829 1830 cmac->nmac_entries = num_mac; 1831 cmac->cntxt_id = ha->hw.rcv_cntxt_id; 1832 1833 for (i = 0; i < num_mac; i++) { 1834 bcopy(mac_addr, cmac->mac_addr[i].addr, Q8_ETHER_ADDR_LEN); 1835 mac_addr = mac_addr + ETHER_ADDR_LEN; 1836 } 1837 1838 if (qla_mbx_cmd(ha, (uint32_t *)cmac, 1839 (sizeof (q80_config_mac_addr_t) >> 2), 1840 ha->hw.mbox, (sizeof(q80_config_mac_addr_rsp_t) >> 2), 1)) { 1841 device_printf(dev, "%s: %s failed0\n", __func__, 1842 (add_mac ? "Add" : "Del")); 1843 return (-1); 1844 } 1845 cmac_rsp = (q80_config_mac_addr_rsp_t *)ha->hw.mbox; 1846 1847 err = Q8_MBX_RSP_STATUS(cmac_rsp->regcnt_status); 1848 1849 if (err) { 1850 device_printf(dev, "%s: %s failed1 [0x%08x]\n", __func__, 1851 (add_mac ? "Add" : "Del"), err); 1852 for (i = 0; i < num_mac; i++) { 1853 device_printf(dev, "%s: %02x:%02x:%02x:%02x:%02x:%02x\n", 1854 __func__, mac_cpy[0], mac_cpy[1], mac_cpy[2], 1855 mac_cpy[3], mac_cpy[4], mac_cpy[5]); 1856 mac_cpy += ETHER_ADDR_LEN; 1857 } 1858 return (-1); 1859 } 1860 1861 return 0; 1862 } 1863 1864 1865 /* 1866 * Name: qla_set_mac_rcv_mode 1867 * Function: Enable/Disable AllMulticast and Promiscous Modes. 1868 */ 1869 static int 1870 qla_set_mac_rcv_mode(qla_host_t *ha, uint32_t mode) 1871 { 1872 q80_config_mac_rcv_mode_t *rcv_mode; 1873 uint32_t err; 1874 q80_config_mac_rcv_mode_rsp_t *rcv_mode_rsp; 1875 device_t dev = ha->pci_dev; 1876 1877 rcv_mode = (q80_config_mac_rcv_mode_t *)ha->hw.mbox; 1878 bzero(rcv_mode, (sizeof (q80_config_mac_rcv_mode_t))); 1879 1880 rcv_mode->opcode = Q8_MBX_CONFIG_MAC_RX_MODE; 1881 rcv_mode->count_version = sizeof (q80_config_mac_rcv_mode_t) >> 2; 1882 rcv_mode->count_version |= Q8_MBX_CMD_VERSION; 1883 1884 rcv_mode->mode = mode; 1885 1886 rcv_mode->cntxt_id = ha->hw.rcv_cntxt_id; 1887 1888 if (qla_mbx_cmd(ha, (uint32_t *)rcv_mode, 1889 (sizeof (q80_config_mac_rcv_mode_t) >> 2), 1890 ha->hw.mbox, (sizeof(q80_config_mac_rcv_mode_rsp_t) >> 2), 1)) { 1891 device_printf(dev, "%s: failed0\n", __func__); 1892 return (-1); 1893 } 1894 rcv_mode_rsp = (q80_config_mac_rcv_mode_rsp_t *)ha->hw.mbox; 1895 1896 err = Q8_MBX_RSP_STATUS(rcv_mode_rsp->regcnt_status); 1897 1898 if (err) { 1899 device_printf(dev, "%s: failed1 [0x%08x]\n", __func__, err); 1900 return (-1); 1901 } 1902 1903 return 0; 1904 } 1905 1906 int 1907 ql_set_promisc(qla_host_t *ha) 1908 { 1909 int ret; 1910 1911 ha->hw.mac_rcv_mode |= Q8_MBX_MAC_RCV_PROMISC_ENABLE; 1912 ret = qla_set_mac_rcv_mode(ha, ha->hw.mac_rcv_mode); 1913 return (ret); 1914 } 1915 1916 void 1917 qla_reset_promisc(qla_host_t *ha) 1918 { 1919 ha->hw.mac_rcv_mode &= ~Q8_MBX_MAC_RCV_PROMISC_ENABLE; 1920 (void)qla_set_mac_rcv_mode(ha, ha->hw.mac_rcv_mode); 1921 } 1922 1923 int 1924 ql_set_allmulti(qla_host_t *ha) 1925 { 1926 int ret; 1927 1928 ha->hw.mac_rcv_mode |= Q8_MBX_MAC_ALL_MULTI_ENABLE; 1929 ret = qla_set_mac_rcv_mode(ha, ha->hw.mac_rcv_mode); 1930 return (ret); 1931 } 1932 1933 void 1934 qla_reset_allmulti(qla_host_t *ha) 1935 { 1936 ha->hw.mac_rcv_mode &= ~Q8_MBX_MAC_ALL_MULTI_ENABLE; 1937 (void)qla_set_mac_rcv_mode(ha, ha->hw.mac_rcv_mode); 1938 } 1939 1940 /* 1941 * Name: ql_set_max_mtu 1942 * Function: 1943 * Sets the maximum transfer unit size for the specified rcv context. 1944 */ 1945 int 1946 ql_set_max_mtu(qla_host_t *ha, uint32_t mtu, uint16_t cntxt_id) 1947 { 1948 device_t dev; 1949 q80_set_max_mtu_t *max_mtu; 1950 q80_set_max_mtu_rsp_t *max_mtu_rsp; 1951 uint32_t err; 1952 1953 dev = ha->pci_dev; 1954 1955 max_mtu = (q80_set_max_mtu_t *)ha->hw.mbox; 1956 bzero(max_mtu, (sizeof (q80_set_max_mtu_t))); 1957 1958 max_mtu->opcode = Q8_MBX_SET_MAX_MTU; 1959 max_mtu->count_version = (sizeof (q80_set_max_mtu_t) >> 2); 1960 max_mtu->count_version |= Q8_MBX_CMD_VERSION; 1961 1962 max_mtu->cntxt_id = cntxt_id; 1963 max_mtu->mtu = mtu; 1964 1965 if (qla_mbx_cmd(ha, (uint32_t *)max_mtu, 1966 (sizeof (q80_set_max_mtu_t) >> 2), 1967 ha->hw.mbox, (sizeof (q80_set_max_mtu_rsp_t) >> 2), 1)) { 1968 device_printf(dev, "%s: failed\n", __func__); 1969 return -1; 1970 } 1971 1972 max_mtu_rsp = (q80_set_max_mtu_rsp_t *)ha->hw.mbox; 1973 1974 err = Q8_MBX_RSP_STATUS(max_mtu_rsp->regcnt_status); 1975 1976 if (err) { 1977 device_printf(dev, "%s: failed [0x%08x]\n", __func__, err); 1978 } 1979 1980 return 0; 1981 } 1982 1983 static int 1984 qla_link_event_req(qla_host_t *ha, uint16_t cntxt_id) 1985 { 1986 device_t dev; 1987 q80_link_event_t *lnk; 1988 q80_link_event_rsp_t *lnk_rsp; 1989 uint32_t err; 1990 1991 dev = ha->pci_dev; 1992 1993 lnk = (q80_link_event_t *)ha->hw.mbox; 1994 bzero(lnk, (sizeof (q80_link_event_t))); 1995 1996 lnk->opcode = Q8_MBX_LINK_EVENT_REQ; 1997 lnk->count_version = (sizeof (q80_link_event_t) >> 2); 1998 lnk->count_version |= Q8_MBX_CMD_VERSION; 1999 2000 lnk->cntxt_id = cntxt_id; 2001 lnk->cmd = Q8_LINK_EVENT_CMD_ENABLE_ASYNC; 2002 2003 if (qla_mbx_cmd(ha, (uint32_t *)lnk, (sizeof (q80_link_event_t) >> 2), 2004 ha->hw.mbox, (sizeof (q80_link_event_rsp_t) >> 2), 0)) { 2005 device_printf(dev, "%s: failed\n", __func__); 2006 return -1; 2007 } 2008 2009 lnk_rsp = (q80_link_event_rsp_t *)ha->hw.mbox; 2010 2011 err = Q8_MBX_RSP_STATUS(lnk_rsp->regcnt_status); 2012 2013 if (err) { 2014 device_printf(dev, "%s: failed [0x%08x]\n", __func__, err); 2015 } 2016 2017 return 0; 2018 } 2019 2020 static int 2021 qla_config_fw_lro(qla_host_t *ha, uint16_t cntxt_id) 2022 { 2023 device_t dev; 2024 q80_config_fw_lro_t *fw_lro; 2025 q80_config_fw_lro_rsp_t *fw_lro_rsp; 2026 uint32_t err; 2027 2028 dev = ha->pci_dev; 2029 2030 fw_lro = (q80_config_fw_lro_t *)ha->hw.mbox; 2031 bzero(fw_lro, sizeof(q80_config_fw_lro_t)); 2032 2033 fw_lro->opcode = Q8_MBX_CONFIG_FW_LRO; 2034 fw_lro->count_version = (sizeof (q80_config_fw_lro_t) >> 2); 2035 fw_lro->count_version |= Q8_MBX_CMD_VERSION; 2036 2037 fw_lro->flags |= Q8_MBX_FW_LRO_IPV4 | Q8_MBX_FW_LRO_IPV4_WO_DST_IP_CHK; 2038 fw_lro->flags |= Q8_MBX_FW_LRO_IPV6 | Q8_MBX_FW_LRO_IPV6_WO_DST_IP_CHK; 2039 2040 fw_lro->cntxt_id = cntxt_id; 2041 2042 if (qla_mbx_cmd(ha, (uint32_t *)fw_lro, 2043 (sizeof (q80_config_fw_lro_t) >> 2), 2044 ha->hw.mbox, (sizeof (q80_config_fw_lro_rsp_t) >> 2), 0)) { 2045 device_printf(dev, "%s: failed\n", __func__); 2046 return -1; 2047 } 2048 2049 fw_lro_rsp = (q80_config_fw_lro_rsp_t *)ha->hw.mbox; 2050 2051 err = Q8_MBX_RSP_STATUS(fw_lro_rsp->regcnt_status); 2052 2053 if (err) { 2054 device_printf(dev, "%s: failed [0x%08x]\n", __func__, err); 2055 } 2056 2057 return 0; 2058 } 2059 2060 static int 2061 qla_set_cam_search_mode(qla_host_t *ha, uint32_t search_mode) 2062 { 2063 device_t dev; 2064 q80_hw_config_t *hw_config; 2065 q80_hw_config_rsp_t *hw_config_rsp; 2066 uint32_t err; 2067 2068 dev = ha->pci_dev; 2069 2070 hw_config = (q80_hw_config_t *)ha->hw.mbox; 2071 bzero(hw_config, sizeof (q80_hw_config_t)); 2072 2073 hw_config->opcode = Q8_MBX_HW_CONFIG; 2074 hw_config->count_version = Q8_HW_CONFIG_SET_CAM_SEARCH_MODE_COUNT; 2075 hw_config->count_version |= Q8_MBX_CMD_VERSION; 2076 2077 hw_config->cmd = Q8_HW_CONFIG_SET_CAM_SEARCH_MODE; 2078 2079 hw_config->u.set_cam_search_mode.mode = search_mode; 2080 2081 if (qla_mbx_cmd(ha, (uint32_t *)hw_config, 2082 (sizeof (q80_hw_config_t) >> 2), 2083 ha->hw.mbox, (sizeof (q80_hw_config_rsp_t) >> 2), 0)) { 2084 device_printf(dev, "%s: failed\n", __func__); 2085 return -1; 2086 } 2087 hw_config_rsp = (q80_hw_config_rsp_t *)ha->hw.mbox; 2088 2089 err = Q8_MBX_RSP_STATUS(hw_config_rsp->regcnt_status); 2090 2091 if (err) { 2092 device_printf(dev, "%s: failed [0x%08x]\n", __func__, err); 2093 } 2094 2095 return 0; 2096 } 2097 2098 static int 2099 qla_get_cam_search_mode(qla_host_t *ha) 2100 { 2101 device_t dev; 2102 q80_hw_config_t *hw_config; 2103 q80_hw_config_rsp_t *hw_config_rsp; 2104 uint32_t err; 2105 2106 dev = ha->pci_dev; 2107 2108 hw_config = (q80_hw_config_t *)ha->hw.mbox; 2109 bzero(hw_config, sizeof (q80_hw_config_t)); 2110 2111 hw_config->opcode = Q8_MBX_HW_CONFIG; 2112 hw_config->count_version = Q8_HW_CONFIG_GET_CAM_SEARCH_MODE_COUNT; 2113 hw_config->count_version |= Q8_MBX_CMD_VERSION; 2114 2115 hw_config->cmd = Q8_HW_CONFIG_GET_CAM_SEARCH_MODE; 2116 2117 if (qla_mbx_cmd(ha, (uint32_t *)hw_config, 2118 (sizeof (q80_hw_config_t) >> 2), 2119 ha->hw.mbox, (sizeof (q80_hw_config_rsp_t) >> 2), 0)) { 2120 device_printf(dev, "%s: failed\n", __func__); 2121 return -1; 2122 } 2123 hw_config_rsp = (q80_hw_config_rsp_t *)ha->hw.mbox; 2124 2125 err = Q8_MBX_RSP_STATUS(hw_config_rsp->regcnt_status); 2126 2127 if (err) { 2128 device_printf(dev, "%s: failed [0x%08x]\n", __func__, err); 2129 } else { 2130 device_printf(dev, "%s: cam search mode [0x%08x]\n", __func__, 2131 hw_config_rsp->u.get_cam_search_mode.mode); 2132 } 2133 2134 return 0; 2135 } 2136 2137 static int 2138 qla_get_hw_stats(qla_host_t *ha, uint32_t cmd, uint32_t rsp_size) 2139 { 2140 device_t dev; 2141 q80_get_stats_t *stat; 2142 q80_get_stats_rsp_t *stat_rsp; 2143 uint32_t err; 2144 2145 dev = ha->pci_dev; 2146 2147 stat = (q80_get_stats_t *)ha->hw.mbox; 2148 bzero(stat, (sizeof (q80_get_stats_t))); 2149 2150 stat->opcode = Q8_MBX_GET_STATS; 2151 stat->count_version = 2; 2152 stat->count_version |= Q8_MBX_CMD_VERSION; 2153 2154 stat->cmd = cmd; 2155 2156 if (qla_mbx_cmd(ha, (uint32_t *)stat, 2, 2157 ha->hw.mbox, (rsp_size >> 2), 0)) { 2158 device_printf(dev, "%s: failed\n", __func__); 2159 return -1; 2160 } 2161 2162 stat_rsp = (q80_get_stats_rsp_t *)ha->hw.mbox; 2163 2164 err = Q8_MBX_RSP_STATUS(stat_rsp->regcnt_status); 2165 2166 if (err) { 2167 return -1; 2168 } 2169 2170 return 0; 2171 } 2172 2173 void 2174 ql_get_stats(qla_host_t *ha) 2175 { 2176 q80_get_stats_rsp_t *stat_rsp; 2177 q80_mac_stats_t *mstat; 2178 q80_xmt_stats_t *xstat; 2179 q80_rcv_stats_t *rstat; 2180 uint32_t cmd; 2181 int i; 2182 struct ifnet *ifp = ha->ifp; 2183 2184 if (ifp == NULL) 2185 return; 2186 2187 if (QLA_LOCK(ha, __func__, QLA_LOCK_DEFAULT_MS_TIMEOUT, 0) != 0) { 2188 device_printf(ha->pci_dev, "%s: failed\n", __func__); 2189 return; 2190 } 2191 2192 if (!(ifp->if_drv_flags & IFF_DRV_RUNNING)) { 2193 QLA_UNLOCK(ha, __func__); 2194 return; 2195 } 2196 2197 stat_rsp = (q80_get_stats_rsp_t *)ha->hw.mbox; 2198 /* 2199 * Get MAC Statistics 2200 */ 2201 cmd = Q8_GET_STATS_CMD_TYPE_MAC; 2202 // cmd |= Q8_GET_STATS_CMD_CLEAR; 2203 2204 cmd |= ((ha->pci_func & 0x1) << 16); 2205 2206 if (ha->qla_watchdog_pause || (!(ifp->if_drv_flags & IFF_DRV_RUNNING)) || 2207 ha->offline) 2208 goto ql_get_stats_exit; 2209 2210 if (qla_get_hw_stats(ha, cmd, sizeof (q80_get_stats_rsp_t)) == 0) { 2211 mstat = (q80_mac_stats_t *)&stat_rsp->u.mac; 2212 bcopy(mstat, &ha->hw.mac, sizeof(q80_mac_stats_t)); 2213 } else { 2214 device_printf(ha->pci_dev, "%s: mac failed [0x%08x]\n", 2215 __func__, ha->hw.mbox[0]); 2216 } 2217 /* 2218 * Get RCV Statistics 2219 */ 2220 cmd = Q8_GET_STATS_CMD_RCV | Q8_GET_STATS_CMD_TYPE_CNTXT; 2221 // cmd |= Q8_GET_STATS_CMD_CLEAR; 2222 cmd |= (ha->hw.rcv_cntxt_id << 16); 2223 2224 if (ha->qla_watchdog_pause || (!(ifp->if_drv_flags & IFF_DRV_RUNNING)) || 2225 ha->offline) 2226 goto ql_get_stats_exit; 2227 2228 if (qla_get_hw_stats(ha, cmd, sizeof (q80_get_stats_rsp_t)) == 0) { 2229 rstat = (q80_rcv_stats_t *)&stat_rsp->u.rcv; 2230 bcopy(rstat, &ha->hw.rcv, sizeof(q80_rcv_stats_t)); 2231 } else { 2232 device_printf(ha->pci_dev, "%s: rcv failed [0x%08x]\n", 2233 __func__, ha->hw.mbox[0]); 2234 } 2235 2236 if (ha->qla_watchdog_pause || (!(ifp->if_drv_flags & IFF_DRV_RUNNING)) || 2237 ha->offline) 2238 goto ql_get_stats_exit; 2239 /* 2240 * Get XMT Statistics 2241 */ 2242 for (i = 0 ; (i < ha->hw.num_tx_rings); i++) { 2243 if (ha->qla_watchdog_pause || 2244 (!(ifp->if_drv_flags & IFF_DRV_RUNNING)) || 2245 ha->offline) 2246 goto ql_get_stats_exit; 2247 2248 cmd = Q8_GET_STATS_CMD_XMT | Q8_GET_STATS_CMD_TYPE_CNTXT; 2249 // cmd |= Q8_GET_STATS_CMD_CLEAR; 2250 cmd |= (ha->hw.tx_cntxt[i].tx_cntxt_id << 16); 2251 2252 if (qla_get_hw_stats(ha, cmd, sizeof(q80_get_stats_rsp_t)) 2253 == 0) { 2254 xstat = (q80_xmt_stats_t *)&stat_rsp->u.xmt; 2255 bcopy(xstat, &ha->hw.xmt[i], sizeof(q80_xmt_stats_t)); 2256 } else { 2257 device_printf(ha->pci_dev, "%s: xmt failed [0x%08x]\n", 2258 __func__, ha->hw.mbox[0]); 2259 } 2260 } 2261 2262 ql_get_stats_exit: 2263 QLA_UNLOCK(ha, __func__); 2264 2265 return; 2266 } 2267 2268 /* 2269 * Name: qla_tx_tso 2270 * Function: Checks if the packet to be transmitted is a candidate for 2271 * Large TCP Segment Offload. If yes, the appropriate fields in the Tx 2272 * Ring Structure are plugged in. 2273 */ 2274 static int 2275 qla_tx_tso(qla_host_t *ha, struct mbuf *mp, q80_tx_cmd_t *tx_cmd, uint8_t *hdr) 2276 { 2277 struct ether_vlan_header *eh; 2278 struct ip *ip = NULL; 2279 struct ip6_hdr *ip6 = NULL; 2280 struct tcphdr *th = NULL; 2281 uint32_t ehdrlen, hdrlen, ip_hlen, tcp_hlen, tcp_opt_off; 2282 uint16_t etype, opcode, offload = 1; 2283 device_t dev; 2284 2285 dev = ha->pci_dev; 2286 2287 2288 eh = mtod(mp, struct ether_vlan_header *); 2289 2290 if (eh->evl_encap_proto == htons(ETHERTYPE_VLAN)) { 2291 ehdrlen = ETHER_HDR_LEN + ETHER_VLAN_ENCAP_LEN; 2292 etype = ntohs(eh->evl_proto); 2293 } else { 2294 ehdrlen = ETHER_HDR_LEN; 2295 etype = ntohs(eh->evl_encap_proto); 2296 } 2297 2298 hdrlen = 0; 2299 2300 switch (etype) { 2301 case ETHERTYPE_IP: 2302 2303 tcp_opt_off = ehdrlen + sizeof(struct ip) + 2304 sizeof(struct tcphdr); 2305 2306 if (mp->m_len < tcp_opt_off) { 2307 m_copydata(mp, 0, tcp_opt_off, hdr); 2308 ip = (struct ip *)(hdr + ehdrlen); 2309 } else { 2310 ip = (struct ip *)(mp->m_data + ehdrlen); 2311 } 2312 2313 ip_hlen = ip->ip_hl << 2; 2314 opcode = Q8_TX_CMD_OP_XMT_TCP_LSO; 2315 2316 2317 if ((ip->ip_p != IPPROTO_TCP) || 2318 (ip_hlen != sizeof (struct ip))){ 2319 /* IP Options are not supported */ 2320 2321 offload = 0; 2322 } else 2323 th = (struct tcphdr *)((caddr_t)ip + ip_hlen); 2324 2325 break; 2326 2327 case ETHERTYPE_IPV6: 2328 2329 tcp_opt_off = ehdrlen + sizeof(struct ip6_hdr) + 2330 sizeof (struct tcphdr); 2331 2332 if (mp->m_len < tcp_opt_off) { 2333 m_copydata(mp, 0, tcp_opt_off, hdr); 2334 ip6 = (struct ip6_hdr *)(hdr + ehdrlen); 2335 } else { 2336 ip6 = (struct ip6_hdr *)(mp->m_data + ehdrlen); 2337 } 2338 2339 ip_hlen = sizeof(struct ip6_hdr); 2340 opcode = Q8_TX_CMD_OP_XMT_TCP_LSO_IPV6; 2341 2342 if (ip6->ip6_nxt != IPPROTO_TCP) { 2343 //device_printf(dev, "%s: ipv6\n", __func__); 2344 offload = 0; 2345 } else 2346 th = (struct tcphdr *)((caddr_t)ip6 + ip_hlen); 2347 break; 2348 2349 default: 2350 QL_DPRINT8(ha, (dev, "%s: type!=ip\n", __func__)); 2351 offload = 0; 2352 break; 2353 } 2354 2355 if (!offload) 2356 return (-1); 2357 2358 tcp_hlen = th->th_off << 2; 2359 hdrlen = ehdrlen + ip_hlen + tcp_hlen; 2360 2361 if (mp->m_len < hdrlen) { 2362 if (mp->m_len < tcp_opt_off) { 2363 if (tcp_hlen > sizeof(struct tcphdr)) { 2364 m_copydata(mp, tcp_opt_off, 2365 (tcp_hlen - sizeof(struct tcphdr)), 2366 &hdr[tcp_opt_off]); 2367 } 2368 } else { 2369 m_copydata(mp, 0, hdrlen, hdr); 2370 } 2371 } 2372 2373 tx_cmd->mss = mp->m_pkthdr.tso_segsz; 2374 2375 tx_cmd->flags_opcode = opcode ; 2376 tx_cmd->tcp_hdr_off = ip_hlen + ehdrlen; 2377 tx_cmd->total_hdr_len = hdrlen; 2378 2379 /* Check for Multicast least significant bit of MSB == 1 */ 2380 if (eh->evl_dhost[0] & 0x01) { 2381 tx_cmd->flags_opcode |= Q8_TX_CMD_FLAGS_MULTICAST; 2382 } 2383 2384 if (mp->m_len < hdrlen) { 2385 printf("%d\n", hdrlen); 2386 return (1); 2387 } 2388 2389 return (0); 2390 } 2391 2392 /* 2393 * Name: qla_tx_chksum 2394 * Function: Checks if the packet to be transmitted is a candidate for 2395 * TCP/UDP Checksum offload. If yes, the appropriate fields in the Tx 2396 * Ring Structure are plugged in. 2397 */ 2398 static int 2399 qla_tx_chksum(qla_host_t *ha, struct mbuf *mp, uint32_t *op_code, 2400 uint32_t *tcp_hdr_off) 2401 { 2402 struct ether_vlan_header *eh; 2403 struct ip *ip; 2404 struct ip6_hdr *ip6; 2405 uint32_t ehdrlen, ip_hlen; 2406 uint16_t etype, opcode, offload = 1; 2407 device_t dev; 2408 uint8_t buf[sizeof(struct ip6_hdr)]; 2409 2410 dev = ha->pci_dev; 2411 2412 *op_code = 0; 2413 2414 if ((mp->m_pkthdr.csum_flags & 2415 (CSUM_TCP|CSUM_UDP|CSUM_TCP_IPV6 | CSUM_UDP_IPV6)) == 0) 2416 return (-1); 2417 2418 eh = mtod(mp, struct ether_vlan_header *); 2419 2420 if (eh->evl_encap_proto == htons(ETHERTYPE_VLAN)) { 2421 ehdrlen = ETHER_HDR_LEN + ETHER_VLAN_ENCAP_LEN; 2422 etype = ntohs(eh->evl_proto); 2423 } else { 2424 ehdrlen = ETHER_HDR_LEN; 2425 etype = ntohs(eh->evl_encap_proto); 2426 } 2427 2428 2429 switch (etype) { 2430 case ETHERTYPE_IP: 2431 ip = (struct ip *)(mp->m_data + ehdrlen); 2432 2433 ip_hlen = sizeof (struct ip); 2434 2435 if (mp->m_len < (ehdrlen + ip_hlen)) { 2436 m_copydata(mp, ehdrlen, sizeof(struct ip), buf); 2437 ip = (struct ip *)buf; 2438 } 2439 2440 if (ip->ip_p == IPPROTO_TCP) 2441 opcode = Q8_TX_CMD_OP_XMT_TCP_CHKSUM; 2442 else if (ip->ip_p == IPPROTO_UDP) 2443 opcode = Q8_TX_CMD_OP_XMT_UDP_CHKSUM; 2444 else { 2445 //device_printf(dev, "%s: ipv4\n", __func__); 2446 offload = 0; 2447 } 2448 break; 2449 2450 case ETHERTYPE_IPV6: 2451 ip6 = (struct ip6_hdr *)(mp->m_data + ehdrlen); 2452 2453 ip_hlen = sizeof(struct ip6_hdr); 2454 2455 if (mp->m_len < (ehdrlen + ip_hlen)) { 2456 m_copydata(mp, ehdrlen, sizeof (struct ip6_hdr), 2457 buf); 2458 ip6 = (struct ip6_hdr *)buf; 2459 } 2460 2461 if (ip6->ip6_nxt == IPPROTO_TCP) 2462 opcode = Q8_TX_CMD_OP_XMT_TCP_CHKSUM_IPV6; 2463 else if (ip6->ip6_nxt == IPPROTO_UDP) 2464 opcode = Q8_TX_CMD_OP_XMT_UDP_CHKSUM_IPV6; 2465 else { 2466 //device_printf(dev, "%s: ipv6\n", __func__); 2467 offload = 0; 2468 } 2469 break; 2470 2471 default: 2472 offload = 0; 2473 break; 2474 } 2475 if (!offload) 2476 return (-1); 2477 2478 *op_code = opcode; 2479 *tcp_hdr_off = (ip_hlen + ehdrlen); 2480 2481 return (0); 2482 } 2483 2484 #define QLA_TX_MIN_FREE 2 2485 /* 2486 * Name: ql_hw_send 2487 * Function: Transmits a packet. It first checks if the packet is a 2488 * candidate for Large TCP Segment Offload and then for UDP/TCP checksum 2489 * offload. If either of these creteria are not met, it is transmitted 2490 * as a regular ethernet frame. 2491 */ 2492 int 2493 ql_hw_send(qla_host_t *ha, bus_dma_segment_t *segs, int nsegs, 2494 uint32_t tx_idx, struct mbuf *mp, uint32_t txr_idx, uint32_t iscsi_pdu) 2495 { 2496 struct ether_vlan_header *eh; 2497 qla_hw_t *hw = &ha->hw; 2498 q80_tx_cmd_t *tx_cmd, tso_cmd; 2499 bus_dma_segment_t *c_seg; 2500 uint32_t num_tx_cmds, hdr_len = 0; 2501 uint32_t total_length = 0, bytes, tx_cmd_count = 0, txr_next; 2502 device_t dev; 2503 int i, ret; 2504 uint8_t *src = NULL, *dst = NULL; 2505 uint8_t frame_hdr[QL_FRAME_HDR_SIZE]; 2506 uint32_t op_code = 0; 2507 uint32_t tcp_hdr_off = 0; 2508 2509 dev = ha->pci_dev; 2510 2511 /* 2512 * Always make sure there is atleast one empty slot in the tx_ring 2513 * tx_ring is considered full when there only one entry available 2514 */ 2515 num_tx_cmds = (nsegs + (Q8_TX_CMD_MAX_SEGMENTS - 1)) >> 2; 2516 2517 total_length = mp->m_pkthdr.len; 2518 if (total_length > QLA_MAX_TSO_FRAME_SIZE) { 2519 device_printf(dev, "%s: total length exceeds maxlen(%d)\n", 2520 __func__, total_length); 2521 return (EINVAL); 2522 } 2523 eh = mtod(mp, struct ether_vlan_header *); 2524 2525 if (mp->m_pkthdr.csum_flags & CSUM_TSO) { 2526 2527 bzero((void *)&tso_cmd, sizeof(q80_tx_cmd_t)); 2528 2529 src = frame_hdr; 2530 ret = qla_tx_tso(ha, mp, &tso_cmd, src); 2531 2532 if (!(ret & ~1)) { 2533 /* find the additional tx_cmd descriptors required */ 2534 2535 if (mp->m_flags & M_VLANTAG) 2536 tso_cmd.total_hdr_len += ETHER_VLAN_ENCAP_LEN; 2537 2538 hdr_len = tso_cmd.total_hdr_len; 2539 2540 bytes = sizeof(q80_tx_cmd_t) - Q8_TX_CMD_TSO_ALIGN; 2541 bytes = QL_MIN(bytes, hdr_len); 2542 2543 num_tx_cmds++; 2544 hdr_len -= bytes; 2545 2546 while (hdr_len) { 2547 bytes = QL_MIN((sizeof(q80_tx_cmd_t)), hdr_len); 2548 hdr_len -= bytes; 2549 num_tx_cmds++; 2550 } 2551 hdr_len = tso_cmd.total_hdr_len; 2552 2553 if (ret == 0) 2554 src = (uint8_t *)eh; 2555 } else 2556 return (EINVAL); 2557 } else { 2558 (void)qla_tx_chksum(ha, mp, &op_code, &tcp_hdr_off); 2559 } 2560 2561 if (hw->tx_cntxt[txr_idx].txr_free <= (num_tx_cmds + QLA_TX_MIN_FREE)) { 2562 ql_hw_tx_done_locked(ha, txr_idx); 2563 if (hw->tx_cntxt[txr_idx].txr_free <= 2564 (num_tx_cmds + QLA_TX_MIN_FREE)) { 2565 QL_DPRINT8(ha, (dev, "%s: (hw->txr_free <= " 2566 "(num_tx_cmds + QLA_TX_MIN_FREE))\n", 2567 __func__)); 2568 return (-1); 2569 } 2570 } 2571 2572 for (i = 0; i < num_tx_cmds; i++) { 2573 int j; 2574 2575 j = (tx_idx+i) & (NUM_TX_DESCRIPTORS - 1); 2576 2577 if (NULL != ha->tx_ring[txr_idx].tx_buf[j].m_head) { 2578 QL_ASSERT(ha, 0, \ 2579 ("%s [%d]: txr_idx = %d tx_idx = %d mbuf = %p\n",\ 2580 __func__, __LINE__, txr_idx, j,\ 2581 ha->tx_ring[txr_idx].tx_buf[j].m_head)); 2582 return (EINVAL); 2583 } 2584 } 2585 2586 tx_cmd = &hw->tx_cntxt[txr_idx].tx_ring_base[tx_idx]; 2587 2588 if (!(mp->m_pkthdr.csum_flags & CSUM_TSO)) { 2589 2590 if (nsegs > ha->hw.max_tx_segs) 2591 ha->hw.max_tx_segs = nsegs; 2592 2593 bzero((void *)tx_cmd, sizeof(q80_tx_cmd_t)); 2594 2595 if (op_code) { 2596 tx_cmd->flags_opcode = op_code; 2597 tx_cmd->tcp_hdr_off = tcp_hdr_off; 2598 2599 } else { 2600 tx_cmd->flags_opcode = Q8_TX_CMD_OP_XMT_ETHER; 2601 } 2602 } else { 2603 bcopy(&tso_cmd, tx_cmd, sizeof(q80_tx_cmd_t)); 2604 ha->tx_tso_frames++; 2605 } 2606 2607 if (eh->evl_encap_proto == htons(ETHERTYPE_VLAN)) { 2608 tx_cmd->flags_opcode |= Q8_TX_CMD_FLAGS_VLAN_TAGGED; 2609 2610 if (iscsi_pdu) 2611 eh->evl_tag |= ha->hw.user_pri_iscsi << 13; 2612 2613 } else if (mp->m_flags & M_VLANTAG) { 2614 2615 if (hdr_len) { /* TSO */ 2616 tx_cmd->flags_opcode |= (Q8_TX_CMD_FLAGS_VLAN_TAGGED | 2617 Q8_TX_CMD_FLAGS_HW_VLAN_ID); 2618 tx_cmd->tcp_hdr_off += ETHER_VLAN_ENCAP_LEN; 2619 } else 2620 tx_cmd->flags_opcode |= Q8_TX_CMD_FLAGS_HW_VLAN_ID; 2621 2622 ha->hw_vlan_tx_frames++; 2623 tx_cmd->vlan_tci = mp->m_pkthdr.ether_vtag; 2624 2625 if (iscsi_pdu) { 2626 tx_cmd->vlan_tci |= ha->hw.user_pri_iscsi << 13; 2627 mp->m_pkthdr.ether_vtag = tx_cmd->vlan_tci; 2628 } 2629 } 2630 2631 2632 tx_cmd->n_bufs = (uint8_t)nsegs; 2633 tx_cmd->data_len_lo = (uint8_t)(total_length & 0xFF); 2634 tx_cmd->data_len_hi = qla_host_to_le16(((uint16_t)(total_length >> 8))); 2635 tx_cmd->cntxtid = Q8_TX_CMD_PORT_CNXTID(ha->pci_func); 2636 2637 c_seg = segs; 2638 2639 while (1) { 2640 for (i = 0; ((i < Q8_TX_CMD_MAX_SEGMENTS) && nsegs); i++) { 2641 2642 switch (i) { 2643 case 0: 2644 tx_cmd->buf1_addr = c_seg->ds_addr; 2645 tx_cmd->buf1_len = c_seg->ds_len; 2646 break; 2647 2648 case 1: 2649 tx_cmd->buf2_addr = c_seg->ds_addr; 2650 tx_cmd->buf2_len = c_seg->ds_len; 2651 break; 2652 2653 case 2: 2654 tx_cmd->buf3_addr = c_seg->ds_addr; 2655 tx_cmd->buf3_len = c_seg->ds_len; 2656 break; 2657 2658 case 3: 2659 tx_cmd->buf4_addr = c_seg->ds_addr; 2660 tx_cmd->buf4_len = c_seg->ds_len; 2661 break; 2662 } 2663 2664 c_seg++; 2665 nsegs--; 2666 } 2667 2668 txr_next = hw->tx_cntxt[txr_idx].txr_next = 2669 (hw->tx_cntxt[txr_idx].txr_next + 1) & 2670 (NUM_TX_DESCRIPTORS - 1); 2671 tx_cmd_count++; 2672 2673 if (!nsegs) 2674 break; 2675 2676 tx_cmd = &hw->tx_cntxt[txr_idx].tx_ring_base[txr_next]; 2677 bzero((void *)tx_cmd, sizeof(q80_tx_cmd_t)); 2678 } 2679 2680 if (mp->m_pkthdr.csum_flags & CSUM_TSO) { 2681 2682 /* TSO : Copy the header in the following tx cmd descriptors */ 2683 2684 txr_next = hw->tx_cntxt[txr_idx].txr_next; 2685 2686 tx_cmd = &hw->tx_cntxt[txr_idx].tx_ring_base[txr_next]; 2687 bzero((void *)tx_cmd, sizeof(q80_tx_cmd_t)); 2688 2689 bytes = sizeof(q80_tx_cmd_t) - Q8_TX_CMD_TSO_ALIGN; 2690 bytes = QL_MIN(bytes, hdr_len); 2691 2692 dst = (uint8_t *)tx_cmd + Q8_TX_CMD_TSO_ALIGN; 2693 2694 if (mp->m_flags & M_VLANTAG) { 2695 /* first copy the src/dst MAC addresses */ 2696 bcopy(src, dst, (ETHER_ADDR_LEN * 2)); 2697 dst += (ETHER_ADDR_LEN * 2); 2698 src += (ETHER_ADDR_LEN * 2); 2699 2700 *((uint16_t *)dst) = htons(ETHERTYPE_VLAN); 2701 dst += 2; 2702 *((uint16_t *)dst) = htons(mp->m_pkthdr.ether_vtag); 2703 dst += 2; 2704 2705 /* bytes left in src header */ 2706 hdr_len -= ((ETHER_ADDR_LEN * 2) + 2707 ETHER_VLAN_ENCAP_LEN); 2708 2709 /* bytes left in TxCmd Entry */ 2710 bytes -= ((ETHER_ADDR_LEN * 2) + ETHER_VLAN_ENCAP_LEN); 2711 2712 2713 bcopy(src, dst, bytes); 2714 src += bytes; 2715 hdr_len -= bytes; 2716 } else { 2717 bcopy(src, dst, bytes); 2718 src += bytes; 2719 hdr_len -= bytes; 2720 } 2721 2722 txr_next = hw->tx_cntxt[txr_idx].txr_next = 2723 (hw->tx_cntxt[txr_idx].txr_next + 1) & 2724 (NUM_TX_DESCRIPTORS - 1); 2725 tx_cmd_count++; 2726 2727 while (hdr_len) { 2728 tx_cmd = &hw->tx_cntxt[txr_idx].tx_ring_base[txr_next]; 2729 bzero((void *)tx_cmd, sizeof(q80_tx_cmd_t)); 2730 2731 bytes = QL_MIN((sizeof(q80_tx_cmd_t)), hdr_len); 2732 2733 bcopy(src, tx_cmd, bytes); 2734 src += bytes; 2735 hdr_len -= bytes; 2736 2737 txr_next = hw->tx_cntxt[txr_idx].txr_next = 2738 (hw->tx_cntxt[txr_idx].txr_next + 1) & 2739 (NUM_TX_DESCRIPTORS - 1); 2740 tx_cmd_count++; 2741 } 2742 } 2743 2744 hw->tx_cntxt[txr_idx].txr_free = 2745 hw->tx_cntxt[txr_idx].txr_free - tx_cmd_count; 2746 2747 QL_UPDATE_TX_PRODUCER_INDEX(ha, hw->tx_cntxt[txr_idx].txr_next,\ 2748 txr_idx); 2749 QL_DPRINT8(ha, (dev, "%s: return\n", __func__)); 2750 2751 return (0); 2752 } 2753 2754 2755 2756 #define Q8_CONFIG_IND_TBL_SIZE 32 /* < Q8_RSS_IND_TBL_SIZE and power of 2 */ 2757 static int 2758 qla_config_rss_ind_table(qla_host_t *ha) 2759 { 2760 uint32_t i, count; 2761 uint8_t rss_ind_tbl[Q8_CONFIG_IND_TBL_SIZE]; 2762 2763 2764 for (i = 0; i < Q8_CONFIG_IND_TBL_SIZE; i++) { 2765 rss_ind_tbl[i] = i % ha->hw.num_sds_rings; 2766 } 2767 2768 for (i = 0; i <= Q8_RSS_IND_TBL_MAX_IDX ; 2769 i = i + Q8_CONFIG_IND_TBL_SIZE) { 2770 2771 if ((i + Q8_CONFIG_IND_TBL_SIZE) > Q8_RSS_IND_TBL_MAX_IDX) { 2772 count = Q8_RSS_IND_TBL_MAX_IDX - i + 1; 2773 } else { 2774 count = Q8_CONFIG_IND_TBL_SIZE; 2775 } 2776 2777 if (qla_set_rss_ind_table(ha, i, count, ha->hw.rcv_cntxt_id, 2778 rss_ind_tbl)) 2779 return (-1); 2780 } 2781 2782 return (0); 2783 } 2784 2785 static int 2786 qla_config_soft_lro(qla_host_t *ha) 2787 { 2788 int i; 2789 qla_hw_t *hw = &ha->hw; 2790 struct lro_ctrl *lro; 2791 2792 for (i = 0; i < hw->num_sds_rings; i++) { 2793 lro = &hw->sds[i].lro; 2794 2795 bzero(lro, sizeof(struct lro_ctrl)); 2796 2797 #if (__FreeBSD_version >= 1100101) 2798 if (tcp_lro_init_args(lro, ha->ifp, 0, NUM_RX_DESCRIPTORS)) { 2799 device_printf(ha->pci_dev, 2800 "%s: tcp_lro_init_args [%d] failed\n", 2801 __func__, i); 2802 return (-1); 2803 } 2804 #else 2805 if (tcp_lro_init(lro)) { 2806 device_printf(ha->pci_dev, 2807 "%s: tcp_lro_init [%d] failed\n", 2808 __func__, i); 2809 return (-1); 2810 } 2811 #endif /* #if (__FreeBSD_version >= 1100101) */ 2812 2813 lro->ifp = ha->ifp; 2814 } 2815 2816 QL_DPRINT2(ha, (ha->pci_dev, "%s: LRO initialized\n", __func__)); 2817 return (0); 2818 } 2819 2820 static void 2821 qla_drain_soft_lro(qla_host_t *ha) 2822 { 2823 int i; 2824 qla_hw_t *hw = &ha->hw; 2825 struct lro_ctrl *lro; 2826 2827 for (i = 0; i < hw->num_sds_rings; i++) { 2828 lro = &hw->sds[i].lro; 2829 2830 #if (__FreeBSD_version >= 1100101) 2831 tcp_lro_flush_all(lro); 2832 #else 2833 struct lro_entry *queued; 2834 2835 while ((!SLIST_EMPTY(&lro->lro_active))) { 2836 queued = SLIST_FIRST(&lro->lro_active); 2837 SLIST_REMOVE_HEAD(&lro->lro_active, next); 2838 tcp_lro_flush(lro, queued); 2839 } 2840 #endif /* #if (__FreeBSD_version >= 1100101) */ 2841 } 2842 2843 return; 2844 } 2845 2846 static void 2847 qla_free_soft_lro(qla_host_t *ha) 2848 { 2849 int i; 2850 qla_hw_t *hw = &ha->hw; 2851 struct lro_ctrl *lro; 2852 2853 for (i = 0; i < hw->num_sds_rings; i++) { 2854 lro = &hw->sds[i].lro; 2855 tcp_lro_free(lro); 2856 } 2857 2858 return; 2859 } 2860 2861 2862 /* 2863 * Name: ql_del_hw_if 2864 * Function: Destroys the hardware specific entities corresponding to an 2865 * Ethernet Interface 2866 */ 2867 void 2868 ql_del_hw_if(qla_host_t *ha) 2869 { 2870 uint32_t i; 2871 uint32_t num_msix; 2872 2873 (void)qla_stop_nic_func(ha); 2874 2875 qla_del_rcv_cntxt(ha); 2876 2877 if(qla_del_xmt_cntxt(ha)) 2878 goto ql_del_hw_if_exit; 2879 2880 if (ha->hw.flags.init_intr_cnxt) { 2881 for (i = 0; i < ha->hw.num_sds_rings; ) { 2882 2883 if ((i + Q8_MAX_INTR_VECTORS) < ha->hw.num_sds_rings) 2884 num_msix = Q8_MAX_INTR_VECTORS; 2885 else 2886 num_msix = ha->hw.num_sds_rings - i; 2887 2888 if (qla_config_intr_cntxt(ha, i, num_msix, 0)) 2889 break; 2890 2891 i += num_msix; 2892 } 2893 2894 ha->hw.flags.init_intr_cnxt = 0; 2895 } 2896 2897 ql_del_hw_if_exit: 2898 if (ha->hw.enable_soft_lro) { 2899 qla_drain_soft_lro(ha); 2900 qla_free_soft_lro(ha); 2901 } 2902 2903 return; 2904 } 2905 2906 void 2907 qla_confirm_9kb_enable(qla_host_t *ha) 2908 { 2909 // uint32_t supports_9kb = 0; 2910 2911 ha->hw.mbx_intr_mask_offset = READ_REG32(ha, Q8_MBOX_INT_MASK_MSIX); 2912 2913 /* Use MSI-X vector 0; Enable Firmware Mailbox Interrupt */ 2914 WRITE_REG32(ha, Q8_MBOX_INT_ENABLE, BIT_2); 2915 WRITE_REG32(ha, ha->hw.mbx_intr_mask_offset, 0x0); 2916 2917 #if 0 2918 qla_get_nic_partition(ha, &supports_9kb, NULL); 2919 2920 if (!supports_9kb) 2921 #endif 2922 ha->hw.enable_9kb = 0; 2923 2924 return; 2925 } 2926 2927 /* 2928 * Name: ql_init_hw_if 2929 * Function: Creates the hardware specific entities corresponding to an 2930 * Ethernet Interface - Transmit and Receive Contexts. Sets the MAC Address 2931 * corresponding to the interface. Enables LRO if allowed. 2932 */ 2933 int 2934 ql_init_hw_if(qla_host_t *ha) 2935 { 2936 device_t dev; 2937 uint32_t i; 2938 uint8_t bcast_mac[6]; 2939 qla_rdesc_t *rdesc; 2940 uint32_t num_msix; 2941 2942 dev = ha->pci_dev; 2943 2944 for (i = 0; i < ha->hw.num_sds_rings; i++) { 2945 bzero(ha->hw.dma_buf.sds_ring[i].dma_b, 2946 ha->hw.dma_buf.sds_ring[i].size); 2947 } 2948 2949 for (i = 0; i < ha->hw.num_sds_rings; ) { 2950 2951 if ((i + Q8_MAX_INTR_VECTORS) < ha->hw.num_sds_rings) 2952 num_msix = Q8_MAX_INTR_VECTORS; 2953 else 2954 num_msix = ha->hw.num_sds_rings - i; 2955 2956 if (qla_config_intr_cntxt(ha, i, num_msix, 1)) { 2957 2958 if (i > 0) { 2959 2960 num_msix = i; 2961 2962 for (i = 0; i < num_msix; ) { 2963 qla_config_intr_cntxt(ha, i, 2964 Q8_MAX_INTR_VECTORS, 0); 2965 i += Q8_MAX_INTR_VECTORS; 2966 } 2967 } 2968 return (-1); 2969 } 2970 2971 i = i + num_msix; 2972 } 2973 2974 ha->hw.flags.init_intr_cnxt = 1; 2975 2976 /* 2977 * Create Receive Context 2978 */ 2979 if (qla_init_rcv_cntxt(ha)) { 2980 return (-1); 2981 } 2982 2983 for (i = 0; i < ha->hw.num_rds_rings; i++) { 2984 rdesc = &ha->hw.rds[i]; 2985 rdesc->rx_next = NUM_RX_DESCRIPTORS - 2; 2986 rdesc->rx_in = 0; 2987 /* Update the RDS Producer Indices */ 2988 QL_UPDATE_RDS_PRODUCER_INDEX(ha, rdesc->prod_std,\ 2989 rdesc->rx_next); 2990 } 2991 2992 /* 2993 * Create Transmit Context 2994 */ 2995 if (qla_init_xmt_cntxt(ha)) { 2996 qla_del_rcv_cntxt(ha); 2997 return (-1); 2998 } 2999 ha->hw.max_tx_segs = 0; 3000 3001 if (qla_config_mac_addr(ha, ha->hw.mac_addr, 1, 1)) 3002 return(-1); 3003 3004 ha->hw.flags.unicast_mac = 1; 3005 3006 bcast_mac[0] = 0xFF; bcast_mac[1] = 0xFF; bcast_mac[2] = 0xFF; 3007 bcast_mac[3] = 0xFF; bcast_mac[4] = 0xFF; bcast_mac[5] = 0xFF; 3008 3009 if (qla_config_mac_addr(ha, bcast_mac, 1, 1)) 3010 return (-1); 3011 3012 ha->hw.flags.bcast_mac = 1; 3013 3014 /* 3015 * program any cached multicast addresses 3016 */ 3017 if (qla_hw_add_all_mcast(ha)) 3018 return (-1); 3019 3020 if (ql_set_max_mtu(ha, ha->max_frame_size, ha->hw.rcv_cntxt_id)) 3021 return (-1); 3022 3023 if (qla_config_rss(ha, ha->hw.rcv_cntxt_id)) 3024 return (-1); 3025 3026 if (qla_config_rss_ind_table(ha)) 3027 return (-1); 3028 3029 if (qla_config_intr_coalesce(ha, ha->hw.rcv_cntxt_id, 0, 1)) 3030 return (-1); 3031 3032 if (qla_link_event_req(ha, ha->hw.rcv_cntxt_id)) 3033 return (-1); 3034 3035 if (ha->ifp->if_capenable & IFCAP_LRO) { 3036 if (ha->hw.enable_hw_lro) { 3037 ha->hw.enable_soft_lro = 0; 3038 3039 if (qla_config_fw_lro(ha, ha->hw.rcv_cntxt_id)) 3040 return (-1); 3041 } else { 3042 ha->hw.enable_soft_lro = 1; 3043 3044 if (qla_config_soft_lro(ha)) 3045 return (-1); 3046 } 3047 } 3048 3049 if (qla_init_nic_func(ha)) 3050 return (-1); 3051 3052 if (qla_query_fw_dcbx_caps(ha)) 3053 return (-1); 3054 3055 for (i = 0; i < ha->hw.num_sds_rings; i++) 3056 QL_ENABLE_INTERRUPTS(ha, i); 3057 3058 return (0); 3059 } 3060 3061 static int 3062 qla_map_sds_to_rds(qla_host_t *ha, uint32_t start_idx, uint32_t num_idx) 3063 { 3064 device_t dev = ha->pci_dev; 3065 q80_rq_map_sds_to_rds_t *map_rings; 3066 q80_rsp_map_sds_to_rds_t *map_rings_rsp; 3067 uint32_t i, err; 3068 qla_hw_t *hw = &ha->hw; 3069 3070 map_rings = (q80_rq_map_sds_to_rds_t *)ha->hw.mbox; 3071 bzero(map_rings, sizeof(q80_rq_map_sds_to_rds_t)); 3072 3073 map_rings->opcode = Q8_MBX_MAP_SDS_TO_RDS; 3074 map_rings->count_version = (sizeof (q80_rq_map_sds_to_rds_t) >> 2); 3075 map_rings->count_version |= Q8_MBX_CMD_VERSION; 3076 3077 map_rings->cntxt_id = hw->rcv_cntxt_id; 3078 map_rings->num_rings = num_idx; 3079 3080 for (i = 0; i < num_idx; i++) { 3081 map_rings->sds_rds[i].sds_ring = i + start_idx; 3082 map_rings->sds_rds[i].rds_ring = i + start_idx; 3083 } 3084 3085 if (qla_mbx_cmd(ha, (uint32_t *)map_rings, 3086 (sizeof (q80_rq_map_sds_to_rds_t) >> 2), 3087 ha->hw.mbox, (sizeof(q80_rsp_add_rcv_rings_t) >> 2), 0)) { 3088 device_printf(dev, "%s: failed0\n", __func__); 3089 return (-1); 3090 } 3091 3092 map_rings_rsp = (q80_rsp_map_sds_to_rds_t *)ha->hw.mbox; 3093 3094 err = Q8_MBX_RSP_STATUS(map_rings_rsp->regcnt_status); 3095 3096 if (err) { 3097 device_printf(dev, "%s: failed1 [0x%08x]\n", __func__, err); 3098 return (-1); 3099 } 3100 3101 return (0); 3102 } 3103 3104 /* 3105 * Name: qla_init_rcv_cntxt 3106 * Function: Creates the Receive Context. 3107 */ 3108 static int 3109 qla_init_rcv_cntxt(qla_host_t *ha) 3110 { 3111 q80_rq_rcv_cntxt_t *rcntxt; 3112 q80_rsp_rcv_cntxt_t *rcntxt_rsp; 3113 q80_stat_desc_t *sdesc; 3114 int i, j; 3115 qla_hw_t *hw = &ha->hw; 3116 device_t dev; 3117 uint32_t err; 3118 uint32_t rcntxt_sds_rings; 3119 uint32_t rcntxt_rds_rings; 3120 uint32_t max_idx; 3121 3122 dev = ha->pci_dev; 3123 3124 /* 3125 * Create Receive Context 3126 */ 3127 3128 for (i = 0; i < hw->num_sds_rings; i++) { 3129 sdesc = (q80_stat_desc_t *)&hw->sds[i].sds_ring_base[0]; 3130 3131 for (j = 0; j < NUM_STATUS_DESCRIPTORS; j++) { 3132 sdesc->data[0] = 1ULL; 3133 sdesc->data[1] = 1ULL; 3134 } 3135 } 3136 3137 rcntxt_sds_rings = hw->num_sds_rings; 3138 if (hw->num_sds_rings > MAX_RCNTXT_SDS_RINGS) 3139 rcntxt_sds_rings = MAX_RCNTXT_SDS_RINGS; 3140 3141 rcntxt_rds_rings = hw->num_rds_rings; 3142 3143 if (hw->num_rds_rings > MAX_RDS_RING_SETS) 3144 rcntxt_rds_rings = MAX_RDS_RING_SETS; 3145 3146 rcntxt = (q80_rq_rcv_cntxt_t *)ha->hw.mbox; 3147 bzero(rcntxt, (sizeof (q80_rq_rcv_cntxt_t))); 3148 3149 rcntxt->opcode = Q8_MBX_CREATE_RX_CNTXT; 3150 rcntxt->count_version = (sizeof (q80_rq_rcv_cntxt_t) >> 2); 3151 rcntxt->count_version |= Q8_MBX_CMD_VERSION; 3152 3153 rcntxt->cap0 = Q8_RCV_CNTXT_CAP0_BASEFW | 3154 Q8_RCV_CNTXT_CAP0_LRO | 3155 Q8_RCV_CNTXT_CAP0_HW_LRO | 3156 Q8_RCV_CNTXT_CAP0_RSS | 3157 Q8_RCV_CNTXT_CAP0_SGL_LRO; 3158 3159 if (ha->hw.enable_9kb) 3160 rcntxt->cap0 |= Q8_RCV_CNTXT_CAP0_SINGLE_JUMBO; 3161 else 3162 rcntxt->cap0 |= Q8_RCV_CNTXT_CAP0_SGL_JUMBO; 3163 3164 if (ha->hw.num_rds_rings > 1) { 3165 rcntxt->nrds_sets_rings = rcntxt_rds_rings | (1 << 5); 3166 rcntxt->cap0 |= Q8_RCV_CNTXT_CAP0_MULTI_RDS; 3167 } else 3168 rcntxt->nrds_sets_rings = 0x1 | (1 << 5); 3169 3170 rcntxt->nsds_rings = rcntxt_sds_rings; 3171 3172 rcntxt->rds_producer_mode = Q8_RCV_CNTXT_RDS_PROD_MODE_UNIQUE; 3173 3174 rcntxt->rcv_vpid = 0; 3175 3176 for (i = 0; i < rcntxt_sds_rings; i++) { 3177 rcntxt->sds[i].paddr = 3178 qla_host_to_le64(hw->dma_buf.sds_ring[i].dma_addr); 3179 rcntxt->sds[i].size = 3180 qla_host_to_le32(NUM_STATUS_DESCRIPTORS); 3181 rcntxt->sds[i].intr_id = qla_host_to_le16(hw->intr_id[i]); 3182 rcntxt->sds[i].intr_src_bit = qla_host_to_le16(0); 3183 } 3184 3185 for (i = 0; i < rcntxt_rds_rings; i++) { 3186 rcntxt->rds[i].paddr_std = 3187 qla_host_to_le64(hw->dma_buf.rds_ring[i].dma_addr); 3188 3189 if (ha->hw.enable_9kb) 3190 rcntxt->rds[i].std_bsize = 3191 qla_host_to_le64(MJUM9BYTES); 3192 else 3193 rcntxt->rds[i].std_bsize = qla_host_to_le64(MCLBYTES); 3194 3195 rcntxt->rds[i].std_nentries = 3196 qla_host_to_le32(NUM_RX_DESCRIPTORS); 3197 } 3198 3199 if (qla_mbx_cmd(ha, (uint32_t *)rcntxt, 3200 (sizeof (q80_rq_rcv_cntxt_t) >> 2), 3201 ha->hw.mbox, (sizeof(q80_rsp_rcv_cntxt_t) >> 2), 0)) { 3202 device_printf(dev, "%s: failed0\n", __func__); 3203 return (-1); 3204 } 3205 3206 rcntxt_rsp = (q80_rsp_rcv_cntxt_t *)ha->hw.mbox; 3207 3208 err = Q8_MBX_RSP_STATUS(rcntxt_rsp->regcnt_status); 3209 3210 if (err) { 3211 device_printf(dev, "%s: failed1 [0x%08x]\n", __func__, err); 3212 return (-1); 3213 } 3214 3215 for (i = 0; i < rcntxt_sds_rings; i++) { 3216 hw->sds[i].sds_consumer = rcntxt_rsp->sds_cons[i]; 3217 } 3218 3219 for (i = 0; i < rcntxt_rds_rings; i++) { 3220 hw->rds[i].prod_std = rcntxt_rsp->rds[i].prod_std; 3221 } 3222 3223 hw->rcv_cntxt_id = rcntxt_rsp->cntxt_id; 3224 3225 ha->hw.flags.init_rx_cnxt = 1; 3226 3227 if (hw->num_sds_rings > MAX_RCNTXT_SDS_RINGS) { 3228 3229 for (i = MAX_RCNTXT_SDS_RINGS; i < hw->num_sds_rings;) { 3230 3231 if ((i + MAX_RCNTXT_SDS_RINGS) < hw->num_sds_rings) 3232 max_idx = MAX_RCNTXT_SDS_RINGS; 3233 else 3234 max_idx = hw->num_sds_rings - i; 3235 3236 err = qla_add_rcv_rings(ha, i, max_idx); 3237 if (err) 3238 return -1; 3239 3240 i += max_idx; 3241 } 3242 } 3243 3244 if (hw->num_rds_rings > 1) { 3245 3246 for (i = 0; i < hw->num_rds_rings; ) { 3247 3248 if ((i + MAX_SDS_TO_RDS_MAP) < hw->num_rds_rings) 3249 max_idx = MAX_SDS_TO_RDS_MAP; 3250 else 3251 max_idx = hw->num_rds_rings - i; 3252 3253 err = qla_map_sds_to_rds(ha, i, max_idx); 3254 if (err) 3255 return -1; 3256 3257 i += max_idx; 3258 } 3259 } 3260 3261 return (0); 3262 } 3263 3264 static int 3265 qla_add_rcv_rings(qla_host_t *ha, uint32_t sds_idx, uint32_t nsds) 3266 { 3267 device_t dev = ha->pci_dev; 3268 q80_rq_add_rcv_rings_t *add_rcv; 3269 q80_rsp_add_rcv_rings_t *add_rcv_rsp; 3270 uint32_t i,j, err; 3271 qla_hw_t *hw = &ha->hw; 3272 3273 add_rcv = (q80_rq_add_rcv_rings_t *)ha->hw.mbox; 3274 bzero(add_rcv, sizeof (q80_rq_add_rcv_rings_t)); 3275 3276 add_rcv->opcode = Q8_MBX_ADD_RX_RINGS; 3277 add_rcv->count_version = (sizeof (q80_rq_add_rcv_rings_t) >> 2); 3278 add_rcv->count_version |= Q8_MBX_CMD_VERSION; 3279 3280 add_rcv->nrds_sets_rings = nsds | (1 << 5); 3281 add_rcv->nsds_rings = nsds; 3282 add_rcv->cntxt_id = hw->rcv_cntxt_id; 3283 3284 for (i = 0; i < nsds; i++) { 3285 3286 j = i + sds_idx; 3287 3288 add_rcv->sds[i].paddr = 3289 qla_host_to_le64(hw->dma_buf.sds_ring[j].dma_addr); 3290 3291 add_rcv->sds[i].size = 3292 qla_host_to_le32(NUM_STATUS_DESCRIPTORS); 3293 3294 add_rcv->sds[i].intr_id = qla_host_to_le16(hw->intr_id[j]); 3295 add_rcv->sds[i].intr_src_bit = qla_host_to_le16(0); 3296 3297 } 3298 3299 for (i = 0; (i < nsds); i++) { 3300 j = i + sds_idx; 3301 3302 add_rcv->rds[i].paddr_std = 3303 qla_host_to_le64(hw->dma_buf.rds_ring[j].dma_addr); 3304 3305 if (ha->hw.enable_9kb) 3306 add_rcv->rds[i].std_bsize = 3307 qla_host_to_le64(MJUM9BYTES); 3308 else 3309 add_rcv->rds[i].std_bsize = qla_host_to_le64(MCLBYTES); 3310 3311 add_rcv->rds[i].std_nentries = 3312 qla_host_to_le32(NUM_RX_DESCRIPTORS); 3313 } 3314 3315 3316 if (qla_mbx_cmd(ha, (uint32_t *)add_rcv, 3317 (sizeof (q80_rq_add_rcv_rings_t) >> 2), 3318 ha->hw.mbox, (sizeof(q80_rsp_add_rcv_rings_t) >> 2), 0)) { 3319 device_printf(dev, "%s: failed0\n", __func__); 3320 return (-1); 3321 } 3322 3323 add_rcv_rsp = (q80_rsp_add_rcv_rings_t *)ha->hw.mbox; 3324 3325 err = Q8_MBX_RSP_STATUS(add_rcv_rsp->regcnt_status); 3326 3327 if (err) { 3328 device_printf(dev, "%s: failed1 [0x%08x]\n", __func__, err); 3329 return (-1); 3330 } 3331 3332 for (i = 0; i < nsds; i++) { 3333 hw->sds[(i + sds_idx)].sds_consumer = add_rcv_rsp->sds_cons[i]; 3334 } 3335 3336 for (i = 0; i < nsds; i++) { 3337 hw->rds[(i + sds_idx)].prod_std = add_rcv_rsp->rds[i].prod_std; 3338 } 3339 3340 return (0); 3341 } 3342 3343 /* 3344 * Name: qla_del_rcv_cntxt 3345 * Function: Destroys the Receive Context. 3346 */ 3347 static void 3348 qla_del_rcv_cntxt(qla_host_t *ha) 3349 { 3350 device_t dev = ha->pci_dev; 3351 q80_rcv_cntxt_destroy_t *rcntxt; 3352 q80_rcv_cntxt_destroy_rsp_t *rcntxt_rsp; 3353 uint32_t err; 3354 uint8_t bcast_mac[6]; 3355 3356 if (!ha->hw.flags.init_rx_cnxt) 3357 return; 3358 3359 if (qla_hw_del_all_mcast(ha)) 3360 return; 3361 3362 if (ha->hw.flags.bcast_mac) { 3363 3364 bcast_mac[0] = 0xFF; bcast_mac[1] = 0xFF; bcast_mac[2] = 0xFF; 3365 bcast_mac[3] = 0xFF; bcast_mac[4] = 0xFF; bcast_mac[5] = 0xFF; 3366 3367 if (qla_config_mac_addr(ha, bcast_mac, 0, 1)) 3368 return; 3369 ha->hw.flags.bcast_mac = 0; 3370 3371 } 3372 3373 if (ha->hw.flags.unicast_mac) { 3374 if (qla_config_mac_addr(ha, ha->hw.mac_addr, 0, 1)) 3375 return; 3376 ha->hw.flags.unicast_mac = 0; 3377 } 3378 3379 rcntxt = (q80_rcv_cntxt_destroy_t *)ha->hw.mbox; 3380 bzero(rcntxt, (sizeof (q80_rcv_cntxt_destroy_t))); 3381 3382 rcntxt->opcode = Q8_MBX_DESTROY_RX_CNTXT; 3383 rcntxt->count_version = (sizeof (q80_rcv_cntxt_destroy_t) >> 2); 3384 rcntxt->count_version |= Q8_MBX_CMD_VERSION; 3385 3386 rcntxt->cntxt_id = ha->hw.rcv_cntxt_id; 3387 3388 if (qla_mbx_cmd(ha, (uint32_t *)rcntxt, 3389 (sizeof (q80_rcv_cntxt_destroy_t) >> 2), 3390 ha->hw.mbox, (sizeof(q80_rcv_cntxt_destroy_rsp_t) >> 2), 0)) { 3391 device_printf(dev, "%s: failed0\n", __func__); 3392 return; 3393 } 3394 rcntxt_rsp = (q80_rcv_cntxt_destroy_rsp_t *)ha->hw.mbox; 3395 3396 err = Q8_MBX_RSP_STATUS(rcntxt_rsp->regcnt_status); 3397 3398 if (err) { 3399 device_printf(dev, "%s: failed1 [0x%08x]\n", __func__, err); 3400 } 3401 3402 ha->hw.flags.init_rx_cnxt = 0; 3403 return; 3404 } 3405 3406 /* 3407 * Name: qla_init_xmt_cntxt 3408 * Function: Creates the Transmit Context. 3409 */ 3410 static int 3411 qla_init_xmt_cntxt_i(qla_host_t *ha, uint32_t txr_idx) 3412 { 3413 device_t dev; 3414 qla_hw_t *hw = &ha->hw; 3415 q80_rq_tx_cntxt_t *tcntxt; 3416 q80_rsp_tx_cntxt_t *tcntxt_rsp; 3417 uint32_t err; 3418 qla_hw_tx_cntxt_t *hw_tx_cntxt; 3419 uint32_t intr_idx; 3420 3421 hw_tx_cntxt = &hw->tx_cntxt[txr_idx]; 3422 3423 dev = ha->pci_dev; 3424 3425 /* 3426 * Create Transmit Context 3427 */ 3428 tcntxt = (q80_rq_tx_cntxt_t *)ha->hw.mbox; 3429 bzero(tcntxt, (sizeof (q80_rq_tx_cntxt_t))); 3430 3431 tcntxt->opcode = Q8_MBX_CREATE_TX_CNTXT; 3432 tcntxt->count_version = (sizeof (q80_rq_tx_cntxt_t) >> 2); 3433 tcntxt->count_version |= Q8_MBX_CMD_VERSION; 3434 3435 intr_idx = txr_idx; 3436 3437 #ifdef QL_ENABLE_ISCSI_TLV 3438 3439 tcntxt->cap0 = Q8_TX_CNTXT_CAP0_BASEFW | Q8_TX_CNTXT_CAP0_LSO | 3440 Q8_TX_CNTXT_CAP0_TC; 3441 3442 if (txr_idx >= (ha->hw.num_tx_rings >> 1)) { 3443 tcntxt->traffic_class = 1; 3444 } 3445 3446 intr_idx = txr_idx % (ha->hw.num_tx_rings >> 1); 3447 3448 #else 3449 tcntxt->cap0 = Q8_TX_CNTXT_CAP0_BASEFW | Q8_TX_CNTXT_CAP0_LSO; 3450 3451 #endif /* #ifdef QL_ENABLE_ISCSI_TLV */ 3452 3453 tcntxt->ntx_rings = 1; 3454 3455 tcntxt->tx_ring[0].paddr = 3456 qla_host_to_le64(hw_tx_cntxt->tx_ring_paddr); 3457 tcntxt->tx_ring[0].tx_consumer = 3458 qla_host_to_le64(hw_tx_cntxt->tx_cons_paddr); 3459 tcntxt->tx_ring[0].nentries = qla_host_to_le16(NUM_TX_DESCRIPTORS); 3460 3461 tcntxt->tx_ring[0].intr_id = qla_host_to_le16(hw->intr_id[intr_idx]); 3462 tcntxt->tx_ring[0].intr_src_bit = qla_host_to_le16(0); 3463 3464 hw_tx_cntxt->txr_free = NUM_TX_DESCRIPTORS; 3465 hw_tx_cntxt->txr_next = hw_tx_cntxt->txr_comp = 0; 3466 *(hw_tx_cntxt->tx_cons) = 0; 3467 3468 if (qla_mbx_cmd(ha, (uint32_t *)tcntxt, 3469 (sizeof (q80_rq_tx_cntxt_t) >> 2), 3470 ha->hw.mbox, 3471 (sizeof(q80_rsp_tx_cntxt_t) >> 2), 0)) { 3472 device_printf(dev, "%s: failed0\n", __func__); 3473 return (-1); 3474 } 3475 tcntxt_rsp = (q80_rsp_tx_cntxt_t *)ha->hw.mbox; 3476 3477 err = Q8_MBX_RSP_STATUS(tcntxt_rsp->regcnt_status); 3478 3479 if (err) { 3480 device_printf(dev, "%s: failed1 [0x%08x]\n", __func__, err); 3481 return -1; 3482 } 3483 3484 hw_tx_cntxt->tx_prod_reg = tcntxt_rsp->tx_ring[0].prod_index; 3485 hw_tx_cntxt->tx_cntxt_id = tcntxt_rsp->tx_ring[0].cntxt_id; 3486 3487 if (qla_config_intr_coalesce(ha, hw_tx_cntxt->tx_cntxt_id, 0, 0)) 3488 return (-1); 3489 3490 return (0); 3491 } 3492 3493 3494 /* 3495 * Name: qla_del_xmt_cntxt 3496 * Function: Destroys the Transmit Context. 3497 */ 3498 static int 3499 qla_del_xmt_cntxt_i(qla_host_t *ha, uint32_t txr_idx) 3500 { 3501 device_t dev = ha->pci_dev; 3502 q80_tx_cntxt_destroy_t *tcntxt; 3503 q80_tx_cntxt_destroy_rsp_t *tcntxt_rsp; 3504 uint32_t err; 3505 3506 tcntxt = (q80_tx_cntxt_destroy_t *)ha->hw.mbox; 3507 bzero(tcntxt, (sizeof (q80_tx_cntxt_destroy_t))); 3508 3509 tcntxt->opcode = Q8_MBX_DESTROY_TX_CNTXT; 3510 tcntxt->count_version = (sizeof (q80_tx_cntxt_destroy_t) >> 2); 3511 tcntxt->count_version |= Q8_MBX_CMD_VERSION; 3512 3513 tcntxt->cntxt_id = ha->hw.tx_cntxt[txr_idx].tx_cntxt_id; 3514 3515 if (qla_mbx_cmd(ha, (uint32_t *)tcntxt, 3516 (sizeof (q80_tx_cntxt_destroy_t) >> 2), 3517 ha->hw.mbox, (sizeof (q80_tx_cntxt_destroy_rsp_t) >> 2), 0)) { 3518 device_printf(dev, "%s: failed0\n", __func__); 3519 return (-1); 3520 } 3521 tcntxt_rsp = (q80_tx_cntxt_destroy_rsp_t *)ha->hw.mbox; 3522 3523 err = Q8_MBX_RSP_STATUS(tcntxt_rsp->regcnt_status); 3524 3525 if (err) { 3526 device_printf(dev, "%s: failed1 [0x%08x]\n", __func__, err); 3527 return (-1); 3528 } 3529 3530 return (0); 3531 } 3532 static int 3533 qla_del_xmt_cntxt(qla_host_t *ha) 3534 { 3535 uint32_t i; 3536 int ret = 0; 3537 3538 if (!ha->hw.flags.init_tx_cnxt) 3539 return (ret); 3540 3541 for (i = 0; i < ha->hw.num_tx_rings; i++) { 3542 if ((ret = qla_del_xmt_cntxt_i(ha, i)) != 0) 3543 break; 3544 } 3545 ha->hw.flags.init_tx_cnxt = 0; 3546 3547 return (ret); 3548 } 3549 3550 static int 3551 qla_init_xmt_cntxt(qla_host_t *ha) 3552 { 3553 uint32_t i, j; 3554 3555 for (i = 0; i < ha->hw.num_tx_rings; i++) { 3556 if (qla_init_xmt_cntxt_i(ha, i) != 0) { 3557 for (j = 0; j < i; j++) { 3558 if (qla_del_xmt_cntxt_i(ha, j)) 3559 break; 3560 } 3561 return (-1); 3562 } 3563 } 3564 ha->hw.flags.init_tx_cnxt = 1; 3565 return (0); 3566 } 3567 3568 static int 3569 qla_hw_all_mcast(qla_host_t *ha, uint32_t add_mcast) 3570 { 3571 int i, nmcast; 3572 uint32_t count = 0; 3573 uint8_t *mcast; 3574 3575 nmcast = ha->hw.nmcast; 3576 3577 QL_DPRINT2(ha, (ha->pci_dev, 3578 "%s:[0x%x] enter nmcast = %d \n", __func__, add_mcast, nmcast)); 3579 3580 mcast = ha->hw.mac_addr_arr; 3581 memset(mcast, 0, (Q8_MAX_MAC_ADDRS * ETHER_ADDR_LEN)); 3582 3583 for (i = 0 ; ((i < Q8_MAX_NUM_MULTICAST_ADDRS) && nmcast); i++) { 3584 if ((ha->hw.mcast[i].addr[0] != 0) || 3585 (ha->hw.mcast[i].addr[1] != 0) || 3586 (ha->hw.mcast[i].addr[2] != 0) || 3587 (ha->hw.mcast[i].addr[3] != 0) || 3588 (ha->hw.mcast[i].addr[4] != 0) || 3589 (ha->hw.mcast[i].addr[5] != 0)) { 3590 3591 bcopy(ha->hw.mcast[i].addr, mcast, ETHER_ADDR_LEN); 3592 mcast = mcast + ETHER_ADDR_LEN; 3593 count++; 3594 3595 device_printf(ha->pci_dev, 3596 "%s: %x:%x:%x:%x:%x:%x \n", 3597 __func__, ha->hw.mcast[i].addr[0], 3598 ha->hw.mcast[i].addr[1], ha->hw.mcast[i].addr[2], 3599 ha->hw.mcast[i].addr[3], ha->hw.mcast[i].addr[4], 3600 ha->hw.mcast[i].addr[5]); 3601 3602 if (count == Q8_MAX_MAC_ADDRS) { 3603 if (qla_config_mac_addr(ha, ha->hw.mac_addr_arr, 3604 add_mcast, count)) { 3605 device_printf(ha->pci_dev, 3606 "%s: failed\n", __func__); 3607 return (-1); 3608 } 3609 3610 count = 0; 3611 mcast = ha->hw.mac_addr_arr; 3612 memset(mcast, 0, 3613 (Q8_MAX_MAC_ADDRS * ETHER_ADDR_LEN)); 3614 } 3615 3616 nmcast--; 3617 } 3618 } 3619 3620 if (count) { 3621 if (qla_config_mac_addr(ha, ha->hw.mac_addr_arr, add_mcast, 3622 count)) { 3623 device_printf(ha->pci_dev, "%s: failed\n", __func__); 3624 return (-1); 3625 } 3626 } 3627 QL_DPRINT2(ha, (ha->pci_dev, 3628 "%s:[0x%x] exit nmcast = %d \n", __func__, add_mcast, nmcast)); 3629 3630 return 0; 3631 } 3632 3633 static int 3634 qla_hw_add_all_mcast(qla_host_t *ha) 3635 { 3636 int ret; 3637 3638 ret = qla_hw_all_mcast(ha, 1); 3639 3640 return (ret); 3641 } 3642 3643 int 3644 qla_hw_del_all_mcast(qla_host_t *ha) 3645 { 3646 int ret; 3647 3648 ret = qla_hw_all_mcast(ha, 0); 3649 3650 bzero(ha->hw.mcast, (sizeof (qla_mcast_t) * Q8_MAX_NUM_MULTICAST_ADDRS)); 3651 ha->hw.nmcast = 0; 3652 3653 return (ret); 3654 } 3655 3656 static int 3657 qla_hw_mac_addr_present(qla_host_t *ha, uint8_t *mta) 3658 { 3659 int i; 3660 3661 for (i = 0; i < Q8_MAX_NUM_MULTICAST_ADDRS; i++) { 3662 if (QL_MAC_CMP(ha->hw.mcast[i].addr, mta) == 0) 3663 return (0); /* its been already added */ 3664 } 3665 return (-1); 3666 } 3667 3668 static int 3669 qla_hw_add_mcast(qla_host_t *ha, uint8_t *mta, uint32_t nmcast) 3670 { 3671 int i; 3672 3673 for (i = 0; i < Q8_MAX_NUM_MULTICAST_ADDRS; i++) { 3674 3675 if ((ha->hw.mcast[i].addr[0] == 0) && 3676 (ha->hw.mcast[i].addr[1] == 0) && 3677 (ha->hw.mcast[i].addr[2] == 0) && 3678 (ha->hw.mcast[i].addr[3] == 0) && 3679 (ha->hw.mcast[i].addr[4] == 0) && 3680 (ha->hw.mcast[i].addr[5] == 0)) { 3681 3682 bcopy(mta, ha->hw.mcast[i].addr, Q8_MAC_ADDR_LEN); 3683 ha->hw.nmcast++; 3684 3685 mta = mta + ETHER_ADDR_LEN; 3686 nmcast--; 3687 3688 if (nmcast == 0) 3689 break; 3690 } 3691 3692 } 3693 return 0; 3694 } 3695 3696 static int 3697 qla_hw_del_mcast(qla_host_t *ha, uint8_t *mta, uint32_t nmcast) 3698 { 3699 int i; 3700 3701 for (i = 0; i < Q8_MAX_NUM_MULTICAST_ADDRS; i++) { 3702 if (QL_MAC_CMP(ha->hw.mcast[i].addr, mta) == 0) { 3703 3704 ha->hw.mcast[i].addr[0] = 0; 3705 ha->hw.mcast[i].addr[1] = 0; 3706 ha->hw.mcast[i].addr[2] = 0; 3707 ha->hw.mcast[i].addr[3] = 0; 3708 ha->hw.mcast[i].addr[4] = 0; 3709 ha->hw.mcast[i].addr[5] = 0; 3710 3711 ha->hw.nmcast--; 3712 3713 mta = mta + ETHER_ADDR_LEN; 3714 nmcast--; 3715 3716 if (nmcast == 0) 3717 break; 3718 } 3719 } 3720 return 0; 3721 } 3722 3723 /* 3724 * Name: ql_hw_set_multi 3725 * Function: Sets the Multicast Addresses provided by the host O.S into the 3726 * hardware (for the given interface) 3727 */ 3728 int 3729 ql_hw_set_multi(qla_host_t *ha, uint8_t *mcast_addr, uint32_t mcnt, 3730 uint32_t add_mac) 3731 { 3732 uint8_t *mta = mcast_addr; 3733 int i; 3734 int ret = 0; 3735 uint32_t count = 0; 3736 uint8_t *mcast; 3737 3738 mcast = ha->hw.mac_addr_arr; 3739 memset(mcast, 0, (Q8_MAX_MAC_ADDRS * ETHER_ADDR_LEN)); 3740 3741 for (i = 0; i < mcnt; i++) { 3742 if (mta[0] || mta[1] || mta[2] || mta[3] || mta[4] || mta[5]) { 3743 if (add_mac) { 3744 if (qla_hw_mac_addr_present(ha, mta) != 0) { 3745 bcopy(mta, mcast, ETHER_ADDR_LEN); 3746 mcast = mcast + ETHER_ADDR_LEN; 3747 count++; 3748 } 3749 } else { 3750 if (qla_hw_mac_addr_present(ha, mta) == 0) { 3751 bcopy(mta, mcast, ETHER_ADDR_LEN); 3752 mcast = mcast + ETHER_ADDR_LEN; 3753 count++; 3754 } 3755 } 3756 } 3757 if (count == Q8_MAX_MAC_ADDRS) { 3758 if (qla_config_mac_addr(ha, ha->hw.mac_addr_arr, 3759 add_mac, count)) { 3760 device_printf(ha->pci_dev, "%s: failed\n", 3761 __func__); 3762 return (-1); 3763 } 3764 3765 if (add_mac) { 3766 qla_hw_add_mcast(ha, ha->hw.mac_addr_arr, 3767 count); 3768 } else { 3769 qla_hw_del_mcast(ha, ha->hw.mac_addr_arr, 3770 count); 3771 } 3772 3773 count = 0; 3774 mcast = ha->hw.mac_addr_arr; 3775 memset(mcast, 0, (Q8_MAX_MAC_ADDRS * ETHER_ADDR_LEN)); 3776 } 3777 3778 mta += Q8_MAC_ADDR_LEN; 3779 } 3780 3781 if (count) { 3782 if (qla_config_mac_addr(ha, ha->hw.mac_addr_arr, add_mac, 3783 count)) { 3784 device_printf(ha->pci_dev, "%s: failed\n", __func__); 3785 return (-1); 3786 } 3787 if (add_mac) { 3788 qla_hw_add_mcast(ha, ha->hw.mac_addr_arr, count); 3789 } else { 3790 qla_hw_del_mcast(ha, ha->hw.mac_addr_arr, count); 3791 } 3792 } 3793 3794 return (ret); 3795 } 3796 3797 /* 3798 * Name: ql_hw_tx_done_locked 3799 * Function: Handle Transmit Completions 3800 */ 3801 void 3802 ql_hw_tx_done_locked(qla_host_t *ha, uint32_t txr_idx) 3803 { 3804 qla_tx_buf_t *txb; 3805 qla_hw_t *hw = &ha->hw; 3806 uint32_t comp_idx, comp_count = 0; 3807 qla_hw_tx_cntxt_t *hw_tx_cntxt; 3808 3809 hw_tx_cntxt = &hw->tx_cntxt[txr_idx]; 3810 3811 /* retrieve index of last entry in tx ring completed */ 3812 comp_idx = qla_le32_to_host(*(hw_tx_cntxt->tx_cons)); 3813 3814 while (comp_idx != hw_tx_cntxt->txr_comp) { 3815 3816 txb = &ha->tx_ring[txr_idx].tx_buf[hw_tx_cntxt->txr_comp]; 3817 3818 hw_tx_cntxt->txr_comp++; 3819 if (hw_tx_cntxt->txr_comp == NUM_TX_DESCRIPTORS) 3820 hw_tx_cntxt->txr_comp = 0; 3821 3822 comp_count++; 3823 3824 if (txb->m_head) { 3825 if_inc_counter(ha->ifp, IFCOUNTER_OPACKETS, 1); 3826 3827 bus_dmamap_sync(ha->tx_tag, txb->map, 3828 BUS_DMASYNC_POSTWRITE); 3829 bus_dmamap_unload(ha->tx_tag, txb->map); 3830 m_freem(txb->m_head); 3831 3832 txb->m_head = NULL; 3833 } 3834 } 3835 3836 hw_tx_cntxt->txr_free += comp_count; 3837 3838 if (hw_tx_cntxt->txr_free > NUM_TX_DESCRIPTORS) 3839 device_printf(ha->pci_dev, "%s [%d]: txr_idx = %d txr_free = %d" 3840 "txr_next = %d txr_comp = %d\n", __func__, __LINE__, 3841 txr_idx, hw_tx_cntxt->txr_free, 3842 hw_tx_cntxt->txr_next, hw_tx_cntxt->txr_comp); 3843 3844 QL_ASSERT(ha, (hw_tx_cntxt->txr_free <= NUM_TX_DESCRIPTORS), \ 3845 ("%s [%d]: txr_idx = %d txr_free = %d txr_next = %d txr_comp = %d\n",\ 3846 __func__, __LINE__, txr_idx, hw_tx_cntxt->txr_free, \ 3847 hw_tx_cntxt->txr_next, hw_tx_cntxt->txr_comp)); 3848 3849 return; 3850 } 3851 3852 void 3853 ql_update_link_state(qla_host_t *ha) 3854 { 3855 uint32_t link_state = 0; 3856 uint32_t prev_link_state; 3857 3858 prev_link_state = ha->hw.link_up; 3859 3860 if (ha->ifp->if_drv_flags & IFF_DRV_RUNNING) { 3861 link_state = READ_REG32(ha, Q8_LINK_STATE); 3862 3863 if (ha->pci_func == 0) { 3864 link_state = (((link_state & 0xF) == 1)? 1 : 0); 3865 } else { 3866 link_state = ((((link_state >> 4)& 0xF) == 1)? 1 : 0); 3867 } 3868 } 3869 3870 atomic_store_rel_8(&ha->hw.link_up, (uint8_t)link_state); 3871 3872 if (prev_link_state != ha->hw.link_up) { 3873 if (ha->hw.link_up) { 3874 if_link_state_change(ha->ifp, LINK_STATE_UP); 3875 } else { 3876 if_link_state_change(ha->ifp, LINK_STATE_DOWN); 3877 } 3878 } 3879 return; 3880 } 3881 3882 int 3883 ql_hw_check_health(qla_host_t *ha) 3884 { 3885 uint32_t val; 3886 3887 ha->hw.health_count++; 3888 3889 if (ha->hw.health_count < 500) 3890 return 0; 3891 3892 ha->hw.health_count = 0; 3893 3894 val = READ_REG32(ha, Q8_ASIC_TEMPERATURE); 3895 3896 if (((val & 0xFFFF) == 2) || ((val & 0xFFFF) == 3) || 3897 (QL_ERR_INJECT(ha, INJCT_TEMPERATURE_FAILURE))) { 3898 device_printf(ha->pci_dev, "%s: Temperature Alert" 3899 " at ts_usecs %ld ts_reg = 0x%08x\n", 3900 __func__, qla_get_usec_timestamp(), val); 3901 3902 if (ha->hw.sp_log_stop_events & Q8_SP_LOG_STOP_TEMP_FAILURE) 3903 ha->hw.sp_log_stop = -1; 3904 3905 QL_INITIATE_RECOVERY(ha); 3906 return -1; 3907 } 3908 3909 val = READ_REG32(ha, Q8_FIRMWARE_HEARTBEAT); 3910 3911 if ((val != ha->hw.hbeat_value) && 3912 (!(QL_ERR_INJECT(ha, INJCT_HEARTBEAT_FAILURE)))) { 3913 ha->hw.hbeat_value = val; 3914 ha->hw.hbeat_failure = 0; 3915 return 0; 3916 } 3917 3918 ha->hw.hbeat_failure++; 3919 3920 3921 if ((ha->dbg_level & 0x8000) && (ha->hw.hbeat_failure == 1)) 3922 device_printf(ha->pci_dev, "%s: Heartbeat Failue 1[0x%08x]\n", 3923 __func__, val); 3924 if (ha->hw.hbeat_failure < 2) /* we ignore the first failure */ 3925 return 0; 3926 else { 3927 uint32_t peg_halt_status1; 3928 uint32_t peg_halt_status2; 3929 3930 peg_halt_status1 = READ_REG32(ha, Q8_PEG_HALT_STATUS1); 3931 peg_halt_status2 = READ_REG32(ha, Q8_PEG_HALT_STATUS2); 3932 3933 device_printf(ha->pci_dev, 3934 "%s: Heartbeat Failue at ts_usecs = %ld " 3935 "fw_heart_beat = 0x%08x " 3936 "peg_halt_status1 = 0x%08x " 3937 "peg_halt_status2 = 0x%08x\n", 3938 __func__, qla_get_usec_timestamp(), val, 3939 peg_halt_status1, peg_halt_status2); 3940 3941 if (ha->hw.sp_log_stop_events & Q8_SP_LOG_STOP_HBEAT_FAILURE) 3942 ha->hw.sp_log_stop = -1; 3943 } 3944 QL_INITIATE_RECOVERY(ha); 3945 3946 return -1; 3947 } 3948 3949 static int 3950 qla_init_nic_func(qla_host_t *ha) 3951 { 3952 device_t dev; 3953 q80_init_nic_func_t *init_nic; 3954 q80_init_nic_func_rsp_t *init_nic_rsp; 3955 uint32_t err; 3956 3957 dev = ha->pci_dev; 3958 3959 init_nic = (q80_init_nic_func_t *)ha->hw.mbox; 3960 bzero(init_nic, sizeof(q80_init_nic_func_t)); 3961 3962 init_nic->opcode = Q8_MBX_INIT_NIC_FUNC; 3963 init_nic->count_version = (sizeof (q80_init_nic_func_t) >> 2); 3964 init_nic->count_version |= Q8_MBX_CMD_VERSION; 3965 3966 init_nic->options = Q8_INIT_NIC_REG_DCBX_CHNG_AEN; 3967 init_nic->options |= Q8_INIT_NIC_REG_SFP_CHNG_AEN; 3968 init_nic->options |= Q8_INIT_NIC_REG_IDC_AEN; 3969 3970 //qla_dump_buf8(ha, __func__, init_nic, sizeof (q80_init_nic_func_t)); 3971 if (qla_mbx_cmd(ha, (uint32_t *)init_nic, 3972 (sizeof (q80_init_nic_func_t) >> 2), 3973 ha->hw.mbox, (sizeof (q80_init_nic_func_rsp_t) >> 2), 0)) { 3974 device_printf(dev, "%s: failed\n", __func__); 3975 return -1; 3976 } 3977 3978 init_nic_rsp = (q80_init_nic_func_rsp_t *)ha->hw.mbox; 3979 // qla_dump_buf8(ha, __func__, init_nic_rsp, sizeof (q80_init_nic_func_rsp_t)); 3980 3981 err = Q8_MBX_RSP_STATUS(init_nic_rsp->regcnt_status); 3982 3983 if (err) { 3984 device_printf(dev, "%s: failed [0x%08x]\n", __func__, err); 3985 } else { 3986 device_printf(dev, "%s: successful\n", __func__); 3987 } 3988 3989 return 0; 3990 } 3991 3992 static int 3993 qla_stop_nic_func(qla_host_t *ha) 3994 { 3995 device_t dev; 3996 q80_stop_nic_func_t *stop_nic; 3997 q80_stop_nic_func_rsp_t *stop_nic_rsp; 3998 uint32_t err; 3999 4000 dev = ha->pci_dev; 4001 4002 stop_nic = (q80_stop_nic_func_t *)ha->hw.mbox; 4003 bzero(stop_nic, sizeof(q80_stop_nic_func_t)); 4004 4005 stop_nic->opcode = Q8_MBX_STOP_NIC_FUNC; 4006 stop_nic->count_version = (sizeof (q80_stop_nic_func_t) >> 2); 4007 stop_nic->count_version |= Q8_MBX_CMD_VERSION; 4008 4009 stop_nic->options = Q8_STOP_NIC_DEREG_DCBX_CHNG_AEN; 4010 stop_nic->options |= Q8_STOP_NIC_DEREG_SFP_CHNG_AEN; 4011 4012 //qla_dump_buf8(ha, __func__, stop_nic, sizeof (q80_stop_nic_func_t)); 4013 if (qla_mbx_cmd(ha, (uint32_t *)stop_nic, 4014 (sizeof (q80_stop_nic_func_t) >> 2), 4015 ha->hw.mbox, (sizeof (q80_stop_nic_func_rsp_t) >> 2), 0)) { 4016 device_printf(dev, "%s: failed\n", __func__); 4017 return -1; 4018 } 4019 4020 stop_nic_rsp = (q80_stop_nic_func_rsp_t *)ha->hw.mbox; 4021 //qla_dump_buf8(ha, __func__, stop_nic_rsp, sizeof (q80_stop_nic_func_rsp_ t)); 4022 4023 err = Q8_MBX_RSP_STATUS(stop_nic_rsp->regcnt_status); 4024 4025 if (err) { 4026 device_printf(dev, "%s: failed [0x%08x]\n", __func__, err); 4027 } 4028 4029 return 0; 4030 } 4031 4032 static int 4033 qla_query_fw_dcbx_caps(qla_host_t *ha) 4034 { 4035 device_t dev; 4036 q80_query_fw_dcbx_caps_t *fw_dcbx; 4037 q80_query_fw_dcbx_caps_rsp_t *fw_dcbx_rsp; 4038 uint32_t err; 4039 4040 dev = ha->pci_dev; 4041 4042 fw_dcbx = (q80_query_fw_dcbx_caps_t *)ha->hw.mbox; 4043 bzero(fw_dcbx, sizeof(q80_query_fw_dcbx_caps_t)); 4044 4045 fw_dcbx->opcode = Q8_MBX_GET_FW_DCBX_CAPS; 4046 fw_dcbx->count_version = (sizeof (q80_query_fw_dcbx_caps_t) >> 2); 4047 fw_dcbx->count_version |= Q8_MBX_CMD_VERSION; 4048 4049 ql_dump_buf8(ha, __func__, fw_dcbx, sizeof (q80_query_fw_dcbx_caps_t)); 4050 if (qla_mbx_cmd(ha, (uint32_t *)fw_dcbx, 4051 (sizeof (q80_query_fw_dcbx_caps_t) >> 2), 4052 ha->hw.mbox, (sizeof (q80_query_fw_dcbx_caps_rsp_t) >> 2), 0)) { 4053 device_printf(dev, "%s: failed\n", __func__); 4054 return -1; 4055 } 4056 4057 fw_dcbx_rsp = (q80_query_fw_dcbx_caps_rsp_t *)ha->hw.mbox; 4058 ql_dump_buf8(ha, __func__, fw_dcbx_rsp, 4059 sizeof (q80_query_fw_dcbx_caps_rsp_t)); 4060 4061 err = Q8_MBX_RSP_STATUS(fw_dcbx_rsp->regcnt_status); 4062 4063 if (err) { 4064 device_printf(dev, "%s: failed [0x%08x]\n", __func__, err); 4065 } 4066 4067 return 0; 4068 } 4069 4070 static int 4071 qla_idc_ack(qla_host_t *ha, uint32_t aen_mb1, uint32_t aen_mb2, 4072 uint32_t aen_mb3, uint32_t aen_mb4) 4073 { 4074 device_t dev; 4075 q80_idc_ack_t *idc_ack; 4076 q80_idc_ack_rsp_t *idc_ack_rsp; 4077 uint32_t err; 4078 int count = 300; 4079 4080 dev = ha->pci_dev; 4081 4082 idc_ack = (q80_idc_ack_t *)ha->hw.mbox; 4083 bzero(idc_ack, sizeof(q80_idc_ack_t)); 4084 4085 idc_ack->opcode = Q8_MBX_IDC_ACK; 4086 idc_ack->count_version = (sizeof (q80_idc_ack_t) >> 2); 4087 idc_ack->count_version |= Q8_MBX_CMD_VERSION; 4088 4089 idc_ack->aen_mb1 = aen_mb1; 4090 idc_ack->aen_mb2 = aen_mb2; 4091 idc_ack->aen_mb3 = aen_mb3; 4092 idc_ack->aen_mb4 = aen_mb4; 4093 4094 ha->hw.imd_compl= 0; 4095 4096 if (qla_mbx_cmd(ha, (uint32_t *)idc_ack, 4097 (sizeof (q80_idc_ack_t) >> 2), 4098 ha->hw.mbox, (sizeof (q80_idc_ack_rsp_t) >> 2), 0)) { 4099 device_printf(dev, "%s: failed\n", __func__); 4100 return -1; 4101 } 4102 4103 idc_ack_rsp = (q80_idc_ack_rsp_t *)ha->hw.mbox; 4104 4105 err = Q8_MBX_RSP_STATUS(idc_ack_rsp->regcnt_status); 4106 4107 if (err) { 4108 device_printf(dev, "%s: failed [0x%08x]\n", __func__, err); 4109 return(-1); 4110 } 4111 4112 while (count && !ha->hw.imd_compl) { 4113 qla_mdelay(__func__, 100); 4114 count--; 4115 } 4116 4117 if (!count) 4118 return -1; 4119 else 4120 device_printf(dev, "%s: count %d\n", __func__, count); 4121 4122 return (0); 4123 } 4124 4125 static int 4126 qla_set_port_config(qla_host_t *ha, uint32_t cfg_bits) 4127 { 4128 device_t dev; 4129 q80_set_port_cfg_t *pcfg; 4130 q80_set_port_cfg_rsp_t *pfg_rsp; 4131 uint32_t err; 4132 int count = 300; 4133 4134 dev = ha->pci_dev; 4135 4136 pcfg = (q80_set_port_cfg_t *)ha->hw.mbox; 4137 bzero(pcfg, sizeof(q80_set_port_cfg_t)); 4138 4139 pcfg->opcode = Q8_MBX_SET_PORT_CONFIG; 4140 pcfg->count_version = (sizeof (q80_set_port_cfg_t) >> 2); 4141 pcfg->count_version |= Q8_MBX_CMD_VERSION; 4142 4143 pcfg->cfg_bits = cfg_bits; 4144 4145 device_printf(dev, "%s: cfg_bits" 4146 " [STD_PAUSE_DIR, PAUSE_TYPE, DCBX]" 4147 " [0x%x, 0x%x, 0x%x]\n", __func__, 4148 ((cfg_bits & Q8_PORT_CFG_BITS_STDPAUSE_DIR_MASK)>>20), 4149 ((cfg_bits & Q8_PORT_CFG_BITS_PAUSE_CFG_MASK) >> 5), 4150 ((cfg_bits & Q8_PORT_CFG_BITS_DCBX_ENABLE) ? 1: 0)); 4151 4152 ha->hw.imd_compl= 0; 4153 4154 if (qla_mbx_cmd(ha, (uint32_t *)pcfg, 4155 (sizeof (q80_set_port_cfg_t) >> 2), 4156 ha->hw.mbox, (sizeof (q80_set_port_cfg_rsp_t) >> 2), 0)) { 4157 device_printf(dev, "%s: failed\n", __func__); 4158 return -1; 4159 } 4160 4161 pfg_rsp = (q80_set_port_cfg_rsp_t *)ha->hw.mbox; 4162 4163 err = Q8_MBX_RSP_STATUS(pfg_rsp->regcnt_status); 4164 4165 if (err == Q8_MBX_RSP_IDC_INTRMD_RSP) { 4166 while (count && !ha->hw.imd_compl) { 4167 qla_mdelay(__func__, 100); 4168 count--; 4169 } 4170 if (count) { 4171 device_printf(dev, "%s: count %d\n", __func__, count); 4172 4173 err = 0; 4174 } 4175 } 4176 4177 if (err) { 4178 device_printf(dev, "%s: failed [0x%08x]\n", __func__, err); 4179 return(-1); 4180 } 4181 4182 return (0); 4183 } 4184 4185 4186 static int 4187 qla_get_minidump_tmplt_size(qla_host_t *ha, uint32_t *size) 4188 { 4189 uint32_t err; 4190 device_t dev = ha->pci_dev; 4191 q80_config_md_templ_size_t *md_size; 4192 q80_config_md_templ_size_rsp_t *md_size_rsp; 4193 4194 #ifndef QL_LDFLASH_FW 4195 4196 ql_minidump_template_hdr_t *hdr; 4197 4198 hdr = (ql_minidump_template_hdr_t *)ql83xx_minidump; 4199 *size = hdr->size_of_template; 4200 return (0); 4201 4202 #endif /* #ifdef QL_LDFLASH_FW */ 4203 4204 md_size = (q80_config_md_templ_size_t *) ha->hw.mbox; 4205 bzero(md_size, sizeof(q80_config_md_templ_size_t)); 4206 4207 md_size->opcode = Q8_MBX_GET_MINIDUMP_TMPLT_SIZE; 4208 md_size->count_version = (sizeof (q80_config_md_templ_size_t) >> 2); 4209 md_size->count_version |= Q8_MBX_CMD_VERSION; 4210 4211 if (qla_mbx_cmd(ha, (uint32_t *) md_size, 4212 (sizeof(q80_config_md_templ_size_t) >> 2), ha->hw.mbox, 4213 (sizeof(q80_config_md_templ_size_rsp_t) >> 2), 0)) { 4214 4215 device_printf(dev, "%s: failed\n", __func__); 4216 4217 return (-1); 4218 } 4219 4220 md_size_rsp = (q80_config_md_templ_size_rsp_t *) ha->hw.mbox; 4221 4222 err = Q8_MBX_RSP_STATUS(md_size_rsp->regcnt_status); 4223 4224 if (err) { 4225 device_printf(dev, "%s: failed [0x%08x]\n", __func__, err); 4226 return(-1); 4227 } 4228 4229 *size = md_size_rsp->templ_size; 4230 4231 return (0); 4232 } 4233 4234 static int 4235 qla_get_port_config(qla_host_t *ha, uint32_t *cfg_bits) 4236 { 4237 device_t dev; 4238 q80_get_port_cfg_t *pcfg; 4239 q80_get_port_cfg_rsp_t *pcfg_rsp; 4240 uint32_t err; 4241 4242 dev = ha->pci_dev; 4243 4244 pcfg = (q80_get_port_cfg_t *)ha->hw.mbox; 4245 bzero(pcfg, sizeof(q80_get_port_cfg_t)); 4246 4247 pcfg->opcode = Q8_MBX_GET_PORT_CONFIG; 4248 pcfg->count_version = (sizeof (q80_get_port_cfg_t) >> 2); 4249 pcfg->count_version |= Q8_MBX_CMD_VERSION; 4250 4251 if (qla_mbx_cmd(ha, (uint32_t *)pcfg, 4252 (sizeof (q80_get_port_cfg_t) >> 2), 4253 ha->hw.mbox, (sizeof (q80_get_port_cfg_rsp_t) >> 2), 0)) { 4254 device_printf(dev, "%s: failed\n", __func__); 4255 return -1; 4256 } 4257 4258 pcfg_rsp = (q80_get_port_cfg_rsp_t *)ha->hw.mbox; 4259 4260 err = Q8_MBX_RSP_STATUS(pcfg_rsp->regcnt_status); 4261 4262 if (err) { 4263 device_printf(dev, "%s: failed [0x%08x]\n", __func__, err); 4264 return(-1); 4265 } 4266 4267 device_printf(dev, "%s: [cfg_bits, port type]" 4268 " [0x%08x, 0x%02x] [STD_PAUSE_DIR, PAUSE_TYPE, DCBX]" 4269 " [0x%x, 0x%x, 0x%x]\n", __func__, 4270 pcfg_rsp->cfg_bits, pcfg_rsp->phys_port_type, 4271 ((pcfg_rsp->cfg_bits & Q8_PORT_CFG_BITS_STDPAUSE_DIR_MASK)>>20), 4272 ((pcfg_rsp->cfg_bits & Q8_PORT_CFG_BITS_PAUSE_CFG_MASK) >> 5), 4273 ((pcfg_rsp->cfg_bits & Q8_PORT_CFG_BITS_DCBX_ENABLE) ? 1: 0) 4274 ); 4275 4276 *cfg_bits = pcfg_rsp->cfg_bits; 4277 4278 return (0); 4279 } 4280 4281 int 4282 ql_iscsi_pdu(qla_host_t *ha, struct mbuf *mp) 4283 { 4284 struct ether_vlan_header *eh; 4285 uint16_t etype; 4286 struct ip *ip = NULL; 4287 struct ip6_hdr *ip6 = NULL; 4288 struct tcphdr *th = NULL; 4289 uint32_t hdrlen; 4290 uint32_t offset; 4291 uint8_t buf[sizeof(struct ip6_hdr)]; 4292 4293 eh = mtod(mp, struct ether_vlan_header *); 4294 4295 if (eh->evl_encap_proto == htons(ETHERTYPE_VLAN)) { 4296 hdrlen = ETHER_HDR_LEN + ETHER_VLAN_ENCAP_LEN; 4297 etype = ntohs(eh->evl_proto); 4298 } else { 4299 hdrlen = ETHER_HDR_LEN; 4300 etype = ntohs(eh->evl_encap_proto); 4301 } 4302 4303 if (etype == ETHERTYPE_IP) { 4304 4305 offset = (hdrlen + sizeof (struct ip)); 4306 4307 if (mp->m_len >= offset) { 4308 ip = (struct ip *)(mp->m_data + hdrlen); 4309 } else { 4310 m_copydata(mp, hdrlen, sizeof (struct ip), buf); 4311 ip = (struct ip *)buf; 4312 } 4313 4314 if (ip->ip_p == IPPROTO_TCP) { 4315 4316 hdrlen += ip->ip_hl << 2; 4317 offset = hdrlen + 4; 4318 4319 if (mp->m_len >= offset) { 4320 th = (struct tcphdr *)(mp->m_data + hdrlen);; 4321 } else { 4322 m_copydata(mp, hdrlen, 4, buf); 4323 th = (struct tcphdr *)buf; 4324 } 4325 } 4326 4327 } else if (etype == ETHERTYPE_IPV6) { 4328 4329 offset = (hdrlen + sizeof (struct ip6_hdr)); 4330 4331 if (mp->m_len >= offset) { 4332 ip6 = (struct ip6_hdr *)(mp->m_data + hdrlen); 4333 } else { 4334 m_copydata(mp, hdrlen, sizeof (struct ip6_hdr), buf); 4335 ip6 = (struct ip6_hdr *)buf; 4336 } 4337 4338 if (ip6->ip6_nxt == IPPROTO_TCP) { 4339 4340 hdrlen += sizeof(struct ip6_hdr); 4341 offset = hdrlen + 4; 4342 4343 if (mp->m_len >= offset) { 4344 th = (struct tcphdr *)(mp->m_data + hdrlen);; 4345 } else { 4346 m_copydata(mp, hdrlen, 4, buf); 4347 th = (struct tcphdr *)buf; 4348 } 4349 } 4350 } 4351 4352 if (th != NULL) { 4353 if ((th->th_sport == htons(3260)) || 4354 (th->th_dport == htons(3260))) 4355 return 0; 4356 } 4357 return (-1); 4358 } 4359 4360 void 4361 qla_hw_async_event(qla_host_t *ha) 4362 { 4363 switch (ha->hw.aen_mb0) { 4364 case 0x8101: 4365 (void)qla_idc_ack(ha, ha->hw.aen_mb1, ha->hw.aen_mb2, 4366 ha->hw.aen_mb3, ha->hw.aen_mb4); 4367 4368 break; 4369 4370 default: 4371 break; 4372 } 4373 4374 return; 4375 } 4376 4377 #ifdef QL_LDFLASH_FW 4378 static int 4379 ql_get_minidump_template(qla_host_t *ha) 4380 { 4381 uint32_t err; 4382 device_t dev = ha->pci_dev; 4383 q80_config_md_templ_cmd_t *md_templ; 4384 q80_config_md_templ_cmd_rsp_t *md_templ_rsp; 4385 4386 md_templ = (q80_config_md_templ_cmd_t *) ha->hw.mbox; 4387 bzero(md_templ, (sizeof (q80_config_md_templ_cmd_t))); 4388 4389 md_templ->opcode = Q8_MBX_GET_MINIDUMP_TMPLT; 4390 md_templ->count_version = ( sizeof(q80_config_md_templ_cmd_t) >> 2); 4391 md_templ->count_version |= Q8_MBX_CMD_VERSION; 4392 4393 md_templ->buf_addr = ha->hw.dma_buf.minidump.dma_addr; 4394 md_templ->buff_size = ha->hw.dma_buf.minidump.size; 4395 4396 if (qla_mbx_cmd(ha, (uint32_t *) md_templ, 4397 (sizeof(q80_config_md_templ_cmd_t) >> 2), 4398 ha->hw.mbox, 4399 (sizeof(q80_config_md_templ_cmd_rsp_t) >> 2), 0)) { 4400 4401 device_printf(dev, "%s: failed\n", __func__); 4402 4403 return (-1); 4404 } 4405 4406 md_templ_rsp = (q80_config_md_templ_cmd_rsp_t *) ha->hw.mbox; 4407 4408 err = Q8_MBX_RSP_STATUS(md_templ_rsp->regcnt_status); 4409 4410 if (err) { 4411 device_printf(dev, "%s: failed [0x%08x]\n", __func__, err); 4412 return (-1); 4413 } 4414 4415 return (0); 4416 4417 } 4418 #endif /* #ifdef QL_LDFLASH_FW */ 4419 4420 /* 4421 * Minidump related functionality 4422 */ 4423 4424 static int ql_parse_template(qla_host_t *ha); 4425 4426 static uint32_t ql_rdcrb(qla_host_t *ha, 4427 ql_minidump_entry_rdcrb_t *crb_entry, 4428 uint32_t * data_buff); 4429 4430 static uint32_t ql_pollrd(qla_host_t *ha, 4431 ql_minidump_entry_pollrd_t *entry, 4432 uint32_t * data_buff); 4433 4434 static uint32_t ql_pollrd_modify_write(qla_host_t *ha, 4435 ql_minidump_entry_rd_modify_wr_with_poll_t *entry, 4436 uint32_t *data_buff); 4437 4438 static uint32_t ql_L2Cache(qla_host_t *ha, 4439 ql_minidump_entry_cache_t *cacheEntry, 4440 uint32_t * data_buff); 4441 4442 static uint32_t ql_L1Cache(qla_host_t *ha, 4443 ql_minidump_entry_cache_t *cacheEntry, 4444 uint32_t *data_buff); 4445 4446 static uint32_t ql_rdocm(qla_host_t *ha, 4447 ql_minidump_entry_rdocm_t *ocmEntry, 4448 uint32_t *data_buff); 4449 4450 static uint32_t ql_rdmem(qla_host_t *ha, 4451 ql_minidump_entry_rdmem_t *mem_entry, 4452 uint32_t *data_buff); 4453 4454 static uint32_t ql_rdrom(qla_host_t *ha, 4455 ql_minidump_entry_rdrom_t *romEntry, 4456 uint32_t *data_buff); 4457 4458 static uint32_t ql_rdmux(qla_host_t *ha, 4459 ql_minidump_entry_mux_t *muxEntry, 4460 uint32_t *data_buff); 4461 4462 static uint32_t ql_rdmux2(qla_host_t *ha, 4463 ql_minidump_entry_mux2_t *muxEntry, 4464 uint32_t *data_buff); 4465 4466 static uint32_t ql_rdqueue(qla_host_t *ha, 4467 ql_minidump_entry_queue_t *queueEntry, 4468 uint32_t *data_buff); 4469 4470 static uint32_t ql_cntrl(qla_host_t *ha, 4471 ql_minidump_template_hdr_t *template_hdr, 4472 ql_minidump_entry_cntrl_t *crbEntry); 4473 4474 4475 static uint32_t 4476 ql_minidump_size(qla_host_t *ha) 4477 { 4478 uint32_t i, k; 4479 uint32_t size = 0; 4480 ql_minidump_template_hdr_t *hdr; 4481 4482 hdr = (ql_minidump_template_hdr_t *)ha->hw.dma_buf.minidump.dma_b; 4483 4484 i = 0x2; 4485 4486 for (k = 1; k < QL_DBG_CAP_SIZE_ARRAY_LEN; k++) { 4487 if (i & ha->hw.mdump_capture_mask) 4488 size += hdr->capture_size_array[k]; 4489 i = i << 1; 4490 } 4491 return (size); 4492 } 4493 4494 static void 4495 ql_free_minidump_buffer(qla_host_t *ha) 4496 { 4497 if (ha->hw.mdump_buffer != NULL) { 4498 free(ha->hw.mdump_buffer, M_QLA83XXBUF); 4499 ha->hw.mdump_buffer = NULL; 4500 ha->hw.mdump_buffer_size = 0; 4501 } 4502 return; 4503 } 4504 4505 static int 4506 ql_alloc_minidump_buffer(qla_host_t *ha) 4507 { 4508 ha->hw.mdump_buffer_size = ql_minidump_size(ha); 4509 4510 if (!ha->hw.mdump_buffer_size) 4511 return (-1); 4512 4513 ha->hw.mdump_buffer = malloc(ha->hw.mdump_buffer_size, M_QLA83XXBUF, 4514 M_NOWAIT); 4515 4516 if (ha->hw.mdump_buffer == NULL) 4517 return (-1); 4518 4519 return (0); 4520 } 4521 4522 static void 4523 ql_free_minidump_template_buffer(qla_host_t *ha) 4524 { 4525 if (ha->hw.mdump_template != NULL) { 4526 free(ha->hw.mdump_template, M_QLA83XXBUF); 4527 ha->hw.mdump_template = NULL; 4528 ha->hw.mdump_template_size = 0; 4529 } 4530 return; 4531 } 4532 4533 static int 4534 ql_alloc_minidump_template_buffer(qla_host_t *ha) 4535 { 4536 ha->hw.mdump_template_size = ha->hw.dma_buf.minidump.size; 4537 4538 ha->hw.mdump_template = malloc(ha->hw.mdump_template_size, 4539 M_QLA83XXBUF, M_NOWAIT); 4540 4541 if (ha->hw.mdump_template == NULL) 4542 return (-1); 4543 4544 return (0); 4545 } 4546 4547 static int 4548 ql_alloc_minidump_buffers(qla_host_t *ha) 4549 { 4550 int ret; 4551 4552 ret = ql_alloc_minidump_template_buffer(ha); 4553 4554 if (ret) 4555 return (ret); 4556 4557 ret = ql_alloc_minidump_buffer(ha); 4558 4559 if (ret) 4560 ql_free_minidump_template_buffer(ha); 4561 4562 return (ret); 4563 } 4564 4565 4566 static uint32_t 4567 ql_validate_minidump_checksum(qla_host_t *ha) 4568 { 4569 uint64_t sum = 0; 4570 int count; 4571 uint32_t *template_buff; 4572 4573 count = ha->hw.dma_buf.minidump.size / sizeof (uint32_t); 4574 template_buff = ha->hw.dma_buf.minidump.dma_b; 4575 4576 while (count-- > 0) { 4577 sum += *template_buff++; 4578 } 4579 4580 while (sum >> 32) { 4581 sum = (sum & 0xFFFFFFFF) + (sum >> 32); 4582 } 4583 4584 return (~sum); 4585 } 4586 4587 int 4588 ql_minidump_init(qla_host_t *ha) 4589 { 4590 int ret = 0; 4591 uint32_t template_size = 0; 4592 device_t dev = ha->pci_dev; 4593 4594 /* 4595 * Get Minidump Template Size 4596 */ 4597 ret = qla_get_minidump_tmplt_size(ha, &template_size); 4598 4599 if (ret || (template_size == 0)) { 4600 device_printf(dev, "%s: failed [%d, %d]\n", __func__, ret, 4601 template_size); 4602 return (-1); 4603 } 4604 4605 /* 4606 * Allocate Memory for Minidump Template 4607 */ 4608 4609 ha->hw.dma_buf.minidump.alignment = 8; 4610 ha->hw.dma_buf.minidump.size = template_size; 4611 4612 #ifdef QL_LDFLASH_FW 4613 if (ql_alloc_dmabuf(ha, &ha->hw.dma_buf.minidump)) { 4614 4615 device_printf(dev, "%s: minidump dma alloc failed\n", __func__); 4616 4617 return (-1); 4618 } 4619 ha->hw.dma_buf.flags.minidump = 1; 4620 4621 /* 4622 * Retrieve Minidump Template 4623 */ 4624 ret = ql_get_minidump_template(ha); 4625 #else 4626 ha->hw.dma_buf.minidump.dma_b = ql83xx_minidump; 4627 4628 #endif /* #ifdef QL_LDFLASH_FW */ 4629 4630 if (ret == 0) { 4631 4632 ret = ql_validate_minidump_checksum(ha); 4633 4634 if (ret == 0) { 4635 4636 ret = ql_alloc_minidump_buffers(ha); 4637 4638 if (ret == 0) 4639 ha->hw.mdump_init = 1; 4640 else 4641 device_printf(dev, 4642 "%s: ql_alloc_minidump_buffers" 4643 " failed\n", __func__); 4644 } else { 4645 device_printf(dev, "%s: ql_validate_minidump_checksum" 4646 " failed\n", __func__); 4647 } 4648 } else { 4649 device_printf(dev, "%s: ql_get_minidump_template failed\n", 4650 __func__); 4651 } 4652 4653 if (ret) 4654 ql_minidump_free(ha); 4655 4656 return (ret); 4657 } 4658 4659 static void 4660 ql_minidump_free(qla_host_t *ha) 4661 { 4662 ha->hw.mdump_init = 0; 4663 if (ha->hw.dma_buf.flags.minidump) { 4664 ha->hw.dma_buf.flags.minidump = 0; 4665 ql_free_dmabuf(ha, &ha->hw.dma_buf.minidump); 4666 } 4667 4668 ql_free_minidump_template_buffer(ha); 4669 ql_free_minidump_buffer(ha); 4670 4671 return; 4672 } 4673 4674 void 4675 ql_minidump(qla_host_t *ha) 4676 { 4677 if (!ha->hw.mdump_init) 4678 return; 4679 4680 if (ha->hw.mdump_done) 4681 return; 4682 ha->hw.mdump_usec_ts = qla_get_usec_timestamp(); 4683 ha->hw.mdump_start_seq_index = ql_stop_sequence(ha); 4684 4685 bzero(ha->hw.mdump_buffer, ha->hw.mdump_buffer_size); 4686 bzero(ha->hw.mdump_template, ha->hw.mdump_template_size); 4687 4688 bcopy(ha->hw.dma_buf.minidump.dma_b, ha->hw.mdump_template, 4689 ha->hw.mdump_template_size); 4690 4691 ql_parse_template(ha); 4692 4693 ql_start_sequence(ha, ha->hw.mdump_start_seq_index); 4694 4695 ha->hw.mdump_done = 1; 4696 4697 return; 4698 } 4699 4700 4701 /* 4702 * helper routines 4703 */ 4704 static void 4705 ql_entry_err_chk(ql_minidump_entry_t *entry, uint32_t esize) 4706 { 4707 if (esize != entry->hdr.entry_capture_size) { 4708 entry->hdr.entry_capture_size = esize; 4709 entry->hdr.driver_flags |= QL_DBG_SIZE_ERR_FLAG; 4710 } 4711 return; 4712 } 4713 4714 4715 static int 4716 ql_parse_template(qla_host_t *ha) 4717 { 4718 uint32_t num_of_entries, buff_level, e_cnt, esize; 4719 uint32_t end_cnt, rv = 0; 4720 char *dump_buff, *dbuff; 4721 int sane_start = 0, sane_end = 0; 4722 ql_minidump_template_hdr_t *template_hdr; 4723 ql_minidump_entry_t *entry; 4724 uint32_t capture_mask; 4725 uint32_t dump_size; 4726 4727 /* Setup parameters */ 4728 template_hdr = (ql_minidump_template_hdr_t *)ha->hw.mdump_template; 4729 4730 if (template_hdr->entry_type == TLHDR) 4731 sane_start = 1; 4732 4733 dump_buff = (char *) ha->hw.mdump_buffer; 4734 4735 num_of_entries = template_hdr->num_of_entries; 4736 4737 entry = (ql_minidump_entry_t *) ((char *)template_hdr 4738 + template_hdr->first_entry_offset ); 4739 4740 template_hdr->saved_state_array[QL_OCM0_ADDR_INDX] = 4741 template_hdr->ocm_window_array[ha->pci_func]; 4742 template_hdr->saved_state_array[QL_PCIE_FUNC_INDX] = ha->pci_func; 4743 4744 capture_mask = ha->hw.mdump_capture_mask; 4745 dump_size = ha->hw.mdump_buffer_size; 4746 4747 template_hdr->driver_capture_mask = capture_mask; 4748 4749 QL_DPRINT80(ha, (ha->pci_dev, 4750 "%s: sane_start = %d num_of_entries = %d " 4751 "capture_mask = 0x%x dump_size = %d \n", 4752 __func__, sane_start, num_of_entries, capture_mask, dump_size)); 4753 4754 for (buff_level = 0, e_cnt = 0; e_cnt < num_of_entries; e_cnt++) { 4755 4756 /* 4757 * If the capture_mask of the entry does not match capture mask 4758 * skip the entry after marking the driver_flags indicator. 4759 */ 4760 4761 if (!(entry->hdr.entry_capture_mask & capture_mask)) { 4762 4763 entry->hdr.driver_flags |= QL_DBG_SKIPPED_FLAG; 4764 entry = (ql_minidump_entry_t *) ((char *) entry 4765 + entry->hdr.entry_size); 4766 continue; 4767 } 4768 4769 /* 4770 * This is ONLY needed in implementations where 4771 * the capture buffer allocated is too small to capture 4772 * all of the required entries for a given capture mask. 4773 * We need to empty the buffer contents to a file 4774 * if possible, before processing the next entry 4775 * If the buff_full_flag is set, no further capture will happen 4776 * and all remaining non-control entries will be skipped. 4777 */ 4778 if (entry->hdr.entry_capture_size != 0) { 4779 if ((buff_level + entry->hdr.entry_capture_size) > 4780 dump_size) { 4781 /* Try to recover by emptying buffer to file */ 4782 entry->hdr.driver_flags |= QL_DBG_SKIPPED_FLAG; 4783 entry = (ql_minidump_entry_t *) ((char *) entry 4784 + entry->hdr.entry_size); 4785 continue; 4786 } 4787 } 4788 4789 /* 4790 * Decode the entry type and process it accordingly 4791 */ 4792 4793 switch (entry->hdr.entry_type) { 4794 case RDNOP: 4795 break; 4796 4797 case RDEND: 4798 if (sane_end == 0) { 4799 end_cnt = e_cnt; 4800 } 4801 sane_end++; 4802 break; 4803 4804 case RDCRB: 4805 dbuff = dump_buff + buff_level; 4806 esize = ql_rdcrb(ha, (void *)entry, (void *)dbuff); 4807 ql_entry_err_chk(entry, esize); 4808 buff_level += esize; 4809 break; 4810 4811 case POLLRD: 4812 dbuff = dump_buff + buff_level; 4813 esize = ql_pollrd(ha, (void *)entry, (void *)dbuff); 4814 ql_entry_err_chk(entry, esize); 4815 buff_level += esize; 4816 break; 4817 4818 case POLLRDMWR: 4819 dbuff = dump_buff + buff_level; 4820 esize = ql_pollrd_modify_write(ha, (void *)entry, 4821 (void *)dbuff); 4822 ql_entry_err_chk(entry, esize); 4823 buff_level += esize; 4824 break; 4825 4826 case L2ITG: 4827 case L2DTG: 4828 case L2DAT: 4829 case L2INS: 4830 dbuff = dump_buff + buff_level; 4831 esize = ql_L2Cache(ha, (void *)entry, (void *)dbuff); 4832 if (esize == -1) { 4833 entry->hdr.driver_flags |= QL_DBG_SKIPPED_FLAG; 4834 } else { 4835 ql_entry_err_chk(entry, esize); 4836 buff_level += esize; 4837 } 4838 break; 4839 4840 case L1DAT: 4841 case L1INS: 4842 dbuff = dump_buff + buff_level; 4843 esize = ql_L1Cache(ha, (void *)entry, (void *)dbuff); 4844 ql_entry_err_chk(entry, esize); 4845 buff_level += esize; 4846 break; 4847 4848 case RDOCM: 4849 dbuff = dump_buff + buff_level; 4850 esize = ql_rdocm(ha, (void *)entry, (void *)dbuff); 4851 ql_entry_err_chk(entry, esize); 4852 buff_level += esize; 4853 break; 4854 4855 case RDMEM: 4856 dbuff = dump_buff + buff_level; 4857 esize = ql_rdmem(ha, (void *)entry, (void *)dbuff); 4858 ql_entry_err_chk(entry, esize); 4859 buff_level += esize; 4860 break; 4861 4862 case BOARD: 4863 case RDROM: 4864 dbuff = dump_buff + buff_level; 4865 esize = ql_rdrom(ha, (void *)entry, (void *)dbuff); 4866 ql_entry_err_chk(entry, esize); 4867 buff_level += esize; 4868 break; 4869 4870 case RDMUX: 4871 dbuff = dump_buff + buff_level; 4872 esize = ql_rdmux(ha, (void *)entry, (void *)dbuff); 4873 ql_entry_err_chk(entry, esize); 4874 buff_level += esize; 4875 break; 4876 4877 case RDMUX2: 4878 dbuff = dump_buff + buff_level; 4879 esize = ql_rdmux2(ha, (void *)entry, (void *)dbuff); 4880 ql_entry_err_chk(entry, esize); 4881 buff_level += esize; 4882 break; 4883 4884 case QUEUE: 4885 dbuff = dump_buff + buff_level; 4886 esize = ql_rdqueue(ha, (void *)entry, (void *)dbuff); 4887 ql_entry_err_chk(entry, esize); 4888 buff_level += esize; 4889 break; 4890 4891 case CNTRL: 4892 if ((rv = ql_cntrl(ha, template_hdr, (void *)entry))) { 4893 entry->hdr.driver_flags |= QL_DBG_SKIPPED_FLAG; 4894 } 4895 break; 4896 default: 4897 entry->hdr.driver_flags |= QL_DBG_SKIPPED_FLAG; 4898 break; 4899 } 4900 /* next entry in the template */ 4901 entry = (ql_minidump_entry_t *) ((char *) entry 4902 + entry->hdr.entry_size); 4903 } 4904 4905 if (!sane_start || (sane_end > 1)) { 4906 device_printf(ha->pci_dev, 4907 "\n%s: Template configuration error. Check Template\n", 4908 __func__); 4909 } 4910 4911 QL_DPRINT80(ha, (ha->pci_dev, "%s: Minidump num of entries = %d\n", 4912 __func__, template_hdr->num_of_entries)); 4913 4914 return 0; 4915 } 4916 4917 /* 4918 * Read CRB operation. 4919 */ 4920 static uint32_t 4921 ql_rdcrb(qla_host_t *ha, ql_minidump_entry_rdcrb_t * crb_entry, 4922 uint32_t * data_buff) 4923 { 4924 int loop_cnt; 4925 int ret; 4926 uint32_t op_count, addr, stride, value = 0; 4927 4928 addr = crb_entry->addr; 4929 op_count = crb_entry->op_count; 4930 stride = crb_entry->addr_stride; 4931 4932 for (loop_cnt = 0; loop_cnt < op_count; loop_cnt++) { 4933 4934 ret = ql_rdwr_indreg32(ha, addr, &value, 1); 4935 4936 if (ret) 4937 return (0); 4938 4939 *data_buff++ = addr; 4940 *data_buff++ = value; 4941 addr = addr + stride; 4942 } 4943 4944 /* 4945 * for testing purpose we return amount of data written 4946 */ 4947 return (op_count * (2 * sizeof(uint32_t))); 4948 } 4949 4950 /* 4951 * Handle L2 Cache. 4952 */ 4953 4954 static uint32_t 4955 ql_L2Cache(qla_host_t *ha, ql_minidump_entry_cache_t *cacheEntry, 4956 uint32_t * data_buff) 4957 { 4958 int i, k; 4959 int loop_cnt; 4960 int ret; 4961 4962 uint32_t read_value; 4963 uint32_t addr, read_addr, cntrl_addr, tag_reg_addr, cntl_value_w; 4964 uint32_t tag_value, read_cnt; 4965 volatile uint8_t cntl_value_r; 4966 long timeout; 4967 uint32_t data; 4968 4969 loop_cnt = cacheEntry->op_count; 4970 4971 read_addr = cacheEntry->read_addr; 4972 cntrl_addr = cacheEntry->control_addr; 4973 cntl_value_w = (uint32_t) cacheEntry->write_value; 4974 4975 tag_reg_addr = cacheEntry->tag_reg_addr; 4976 4977 tag_value = cacheEntry->init_tag_value; 4978 read_cnt = cacheEntry->read_addr_cnt; 4979 4980 for (i = 0; i < loop_cnt; i++) { 4981 4982 ret = ql_rdwr_indreg32(ha, tag_reg_addr, &tag_value, 0); 4983 if (ret) 4984 return (0); 4985 4986 if (cacheEntry->write_value != 0) { 4987 4988 ret = ql_rdwr_indreg32(ha, cntrl_addr, 4989 &cntl_value_w, 0); 4990 if (ret) 4991 return (0); 4992 } 4993 4994 if (cacheEntry->poll_mask != 0) { 4995 4996 timeout = cacheEntry->poll_wait; 4997 4998 ret = ql_rdwr_indreg32(ha, cntrl_addr, &data, 1); 4999 if (ret) 5000 return (0); 5001 5002 cntl_value_r = (uint8_t)data; 5003 5004 while ((cntl_value_r & cacheEntry->poll_mask) != 0) { 5005 5006 if (timeout) { 5007 qla_mdelay(__func__, 1); 5008 timeout--; 5009 } else 5010 break; 5011 5012 ret = ql_rdwr_indreg32(ha, cntrl_addr, 5013 &data, 1); 5014 if (ret) 5015 return (0); 5016 5017 cntl_value_r = (uint8_t)data; 5018 } 5019 if (!timeout) { 5020 /* Report timeout error. 5021 * core dump capture failed 5022 * Skip remaining entries. 5023 * Write buffer out to file 5024 * Use driver specific fields in template header 5025 * to report this error. 5026 */ 5027 return (-1); 5028 } 5029 } 5030 5031 addr = read_addr; 5032 for (k = 0; k < read_cnt; k++) { 5033 5034 ret = ql_rdwr_indreg32(ha, addr, &read_value, 1); 5035 if (ret) 5036 return (0); 5037 5038 *data_buff++ = read_value; 5039 addr += cacheEntry->read_addr_stride; 5040 } 5041 5042 tag_value += cacheEntry->tag_value_stride; 5043 } 5044 5045 return (read_cnt * loop_cnt * sizeof(uint32_t)); 5046 } 5047 5048 /* 5049 * Handle L1 Cache. 5050 */ 5051 5052 static uint32_t 5053 ql_L1Cache(qla_host_t *ha, 5054 ql_minidump_entry_cache_t *cacheEntry, 5055 uint32_t *data_buff) 5056 { 5057 int ret; 5058 int i, k; 5059 int loop_cnt; 5060 5061 uint32_t read_value; 5062 uint32_t addr, read_addr, cntrl_addr, tag_reg_addr; 5063 uint32_t tag_value, read_cnt; 5064 uint32_t cntl_value_w; 5065 5066 loop_cnt = cacheEntry->op_count; 5067 5068 read_addr = cacheEntry->read_addr; 5069 cntrl_addr = cacheEntry->control_addr; 5070 cntl_value_w = (uint32_t) cacheEntry->write_value; 5071 5072 tag_reg_addr = cacheEntry->tag_reg_addr; 5073 5074 tag_value = cacheEntry->init_tag_value; 5075 read_cnt = cacheEntry->read_addr_cnt; 5076 5077 for (i = 0; i < loop_cnt; i++) { 5078 5079 ret = ql_rdwr_indreg32(ha, tag_reg_addr, &tag_value, 0); 5080 if (ret) 5081 return (0); 5082 5083 ret = ql_rdwr_indreg32(ha, cntrl_addr, &cntl_value_w, 0); 5084 if (ret) 5085 return (0); 5086 5087 addr = read_addr; 5088 for (k = 0; k < read_cnt; k++) { 5089 5090 ret = ql_rdwr_indreg32(ha, addr, &read_value, 1); 5091 if (ret) 5092 return (0); 5093 5094 *data_buff++ = read_value; 5095 addr += cacheEntry->read_addr_stride; 5096 } 5097 5098 tag_value += cacheEntry->tag_value_stride; 5099 } 5100 5101 return (read_cnt * loop_cnt * sizeof(uint32_t)); 5102 } 5103 5104 /* 5105 * Reading OCM memory 5106 */ 5107 5108 static uint32_t 5109 ql_rdocm(qla_host_t *ha, 5110 ql_minidump_entry_rdocm_t *ocmEntry, 5111 uint32_t *data_buff) 5112 { 5113 int i, loop_cnt; 5114 volatile uint32_t addr; 5115 volatile uint32_t value; 5116 5117 addr = ocmEntry->read_addr; 5118 loop_cnt = ocmEntry->op_count; 5119 5120 for (i = 0; i < loop_cnt; i++) { 5121 value = READ_REG32(ha, addr); 5122 *data_buff++ = value; 5123 addr += ocmEntry->read_addr_stride; 5124 } 5125 return (loop_cnt * sizeof(value)); 5126 } 5127 5128 /* 5129 * Read memory 5130 */ 5131 5132 static uint32_t 5133 ql_rdmem(qla_host_t *ha, 5134 ql_minidump_entry_rdmem_t *mem_entry, 5135 uint32_t *data_buff) 5136 { 5137 int ret; 5138 int i, loop_cnt; 5139 volatile uint32_t addr; 5140 q80_offchip_mem_val_t val; 5141 5142 addr = mem_entry->read_addr; 5143 5144 /* size in bytes / 16 */ 5145 loop_cnt = mem_entry->read_data_size / (sizeof(uint32_t) * 4); 5146 5147 for (i = 0; i < loop_cnt; i++) { 5148 5149 ret = ql_rdwr_offchip_mem(ha, (addr & 0x0ffffffff), &val, 1); 5150 if (ret) 5151 return (0); 5152 5153 *data_buff++ = val.data_lo; 5154 *data_buff++ = val.data_hi; 5155 *data_buff++ = val.data_ulo; 5156 *data_buff++ = val.data_uhi; 5157 5158 addr += (sizeof(uint32_t) * 4); 5159 } 5160 5161 return (loop_cnt * (sizeof(uint32_t) * 4)); 5162 } 5163 5164 /* 5165 * Read Rom 5166 */ 5167 5168 static uint32_t 5169 ql_rdrom(qla_host_t *ha, 5170 ql_minidump_entry_rdrom_t *romEntry, 5171 uint32_t *data_buff) 5172 { 5173 int ret; 5174 int i, loop_cnt; 5175 uint32_t addr; 5176 uint32_t value; 5177 5178 addr = romEntry->read_addr; 5179 loop_cnt = romEntry->read_data_size; /* This is size in bytes */ 5180 loop_cnt /= sizeof(value); 5181 5182 for (i = 0; i < loop_cnt; i++) { 5183 5184 ret = ql_rd_flash32(ha, addr, &value); 5185 if (ret) 5186 return (0); 5187 5188 *data_buff++ = value; 5189 addr += sizeof(value); 5190 } 5191 5192 return (loop_cnt * sizeof(value)); 5193 } 5194 5195 /* 5196 * Read MUX data 5197 */ 5198 5199 static uint32_t 5200 ql_rdmux(qla_host_t *ha, 5201 ql_minidump_entry_mux_t *muxEntry, 5202 uint32_t *data_buff) 5203 { 5204 int ret; 5205 int loop_cnt; 5206 uint32_t read_value, sel_value; 5207 uint32_t read_addr, select_addr; 5208 5209 select_addr = muxEntry->select_addr; 5210 sel_value = muxEntry->select_value; 5211 read_addr = muxEntry->read_addr; 5212 5213 for (loop_cnt = 0; loop_cnt < muxEntry->op_count; loop_cnt++) { 5214 5215 ret = ql_rdwr_indreg32(ha, select_addr, &sel_value, 0); 5216 if (ret) 5217 return (0); 5218 5219 ret = ql_rdwr_indreg32(ha, read_addr, &read_value, 1); 5220 if (ret) 5221 return (0); 5222 5223 *data_buff++ = sel_value; 5224 *data_buff++ = read_value; 5225 5226 sel_value += muxEntry->select_value_stride; 5227 } 5228 5229 return (loop_cnt * (2 * sizeof(uint32_t))); 5230 } 5231 5232 static uint32_t 5233 ql_rdmux2(qla_host_t *ha, 5234 ql_minidump_entry_mux2_t *muxEntry, 5235 uint32_t *data_buff) 5236 { 5237 int ret; 5238 int loop_cnt; 5239 5240 uint32_t select_addr_1, select_addr_2; 5241 uint32_t select_value_1, select_value_2; 5242 uint32_t select_value_count, select_value_mask; 5243 uint32_t read_addr, read_value; 5244 5245 select_addr_1 = muxEntry->select_addr_1; 5246 select_addr_2 = muxEntry->select_addr_2; 5247 select_value_1 = muxEntry->select_value_1; 5248 select_value_2 = muxEntry->select_value_2; 5249 select_value_count = muxEntry->select_value_count; 5250 select_value_mask = muxEntry->select_value_mask; 5251 5252 read_addr = muxEntry->read_addr; 5253 5254 for (loop_cnt = 0; loop_cnt < muxEntry->select_value_count; 5255 loop_cnt++) { 5256 5257 uint32_t temp_sel_val; 5258 5259 ret = ql_rdwr_indreg32(ha, select_addr_1, &select_value_1, 0); 5260 if (ret) 5261 return (0); 5262 5263 temp_sel_val = select_value_1 & select_value_mask; 5264 5265 ret = ql_rdwr_indreg32(ha, select_addr_2, &temp_sel_val, 0); 5266 if (ret) 5267 return (0); 5268 5269 ret = ql_rdwr_indreg32(ha, read_addr, &read_value, 1); 5270 if (ret) 5271 return (0); 5272 5273 *data_buff++ = temp_sel_val; 5274 *data_buff++ = read_value; 5275 5276 ret = ql_rdwr_indreg32(ha, select_addr_1, &select_value_2, 0); 5277 if (ret) 5278 return (0); 5279 5280 temp_sel_val = select_value_2 & select_value_mask; 5281 5282 ret = ql_rdwr_indreg32(ha, select_addr_2, &temp_sel_val, 0); 5283 if (ret) 5284 return (0); 5285 5286 ret = ql_rdwr_indreg32(ha, read_addr, &read_value, 1); 5287 if (ret) 5288 return (0); 5289 5290 *data_buff++ = temp_sel_val; 5291 *data_buff++ = read_value; 5292 5293 select_value_1 += muxEntry->select_value_stride; 5294 select_value_2 += muxEntry->select_value_stride; 5295 } 5296 5297 return (loop_cnt * (4 * sizeof(uint32_t))); 5298 } 5299 5300 /* 5301 * Handling Queue State Reads. 5302 */ 5303 5304 static uint32_t 5305 ql_rdqueue(qla_host_t *ha, 5306 ql_minidump_entry_queue_t *queueEntry, 5307 uint32_t *data_buff) 5308 { 5309 int ret; 5310 int loop_cnt, k; 5311 uint32_t read_value; 5312 uint32_t read_addr, read_stride, select_addr; 5313 uint32_t queue_id, read_cnt; 5314 5315 read_cnt = queueEntry->read_addr_cnt; 5316 read_stride = queueEntry->read_addr_stride; 5317 select_addr = queueEntry->select_addr; 5318 5319 for (loop_cnt = 0, queue_id = 0; loop_cnt < queueEntry->op_count; 5320 loop_cnt++) { 5321 5322 ret = ql_rdwr_indreg32(ha, select_addr, &queue_id, 0); 5323 if (ret) 5324 return (0); 5325 5326 read_addr = queueEntry->read_addr; 5327 5328 for (k = 0; k < read_cnt; k++) { 5329 5330 ret = ql_rdwr_indreg32(ha, read_addr, &read_value, 1); 5331 if (ret) 5332 return (0); 5333 5334 *data_buff++ = read_value; 5335 read_addr += read_stride; 5336 } 5337 5338 queue_id += queueEntry->queue_id_stride; 5339 } 5340 5341 return (loop_cnt * (read_cnt * sizeof(uint32_t))); 5342 } 5343 5344 /* 5345 * Handling control entries. 5346 */ 5347 5348 static uint32_t 5349 ql_cntrl(qla_host_t *ha, 5350 ql_minidump_template_hdr_t *template_hdr, 5351 ql_minidump_entry_cntrl_t *crbEntry) 5352 { 5353 int ret; 5354 int count; 5355 uint32_t opcode, read_value, addr, entry_addr; 5356 long timeout; 5357 5358 entry_addr = crbEntry->addr; 5359 5360 for (count = 0; count < crbEntry->op_count; count++) { 5361 opcode = crbEntry->opcode; 5362 5363 if (opcode & QL_DBG_OPCODE_WR) { 5364 5365 ret = ql_rdwr_indreg32(ha, entry_addr, 5366 &crbEntry->value_1, 0); 5367 if (ret) 5368 return (0); 5369 5370 opcode &= ~QL_DBG_OPCODE_WR; 5371 } 5372 5373 if (opcode & QL_DBG_OPCODE_RW) { 5374 5375 ret = ql_rdwr_indreg32(ha, entry_addr, &read_value, 1); 5376 if (ret) 5377 return (0); 5378 5379 ret = ql_rdwr_indreg32(ha, entry_addr, &read_value, 0); 5380 if (ret) 5381 return (0); 5382 5383 opcode &= ~QL_DBG_OPCODE_RW; 5384 } 5385 5386 if (opcode & QL_DBG_OPCODE_AND) { 5387 5388 ret = ql_rdwr_indreg32(ha, entry_addr, &read_value, 1); 5389 if (ret) 5390 return (0); 5391 5392 read_value &= crbEntry->value_2; 5393 opcode &= ~QL_DBG_OPCODE_AND; 5394 5395 if (opcode & QL_DBG_OPCODE_OR) { 5396 read_value |= crbEntry->value_3; 5397 opcode &= ~QL_DBG_OPCODE_OR; 5398 } 5399 5400 ret = ql_rdwr_indreg32(ha, entry_addr, &read_value, 0); 5401 if (ret) 5402 return (0); 5403 } 5404 5405 if (opcode & QL_DBG_OPCODE_OR) { 5406 5407 ret = ql_rdwr_indreg32(ha, entry_addr, &read_value, 1); 5408 if (ret) 5409 return (0); 5410 5411 read_value |= crbEntry->value_3; 5412 5413 ret = ql_rdwr_indreg32(ha, entry_addr, &read_value, 0); 5414 if (ret) 5415 return (0); 5416 5417 opcode &= ~QL_DBG_OPCODE_OR; 5418 } 5419 5420 if (opcode & QL_DBG_OPCODE_POLL) { 5421 5422 opcode &= ~QL_DBG_OPCODE_POLL; 5423 timeout = crbEntry->poll_timeout; 5424 addr = entry_addr; 5425 5426 ret = ql_rdwr_indreg32(ha, addr, &read_value, 1); 5427 if (ret) 5428 return (0); 5429 5430 while ((read_value & crbEntry->value_2) 5431 != crbEntry->value_1) { 5432 5433 if (timeout) { 5434 qla_mdelay(__func__, 1); 5435 timeout--; 5436 } else 5437 break; 5438 5439 ret = ql_rdwr_indreg32(ha, addr, 5440 &read_value, 1); 5441 if (ret) 5442 return (0); 5443 } 5444 5445 if (!timeout) { 5446 /* 5447 * Report timeout error. 5448 * core dump capture failed 5449 * Skip remaining entries. 5450 * Write buffer out to file 5451 * Use driver specific fields in template header 5452 * to report this error. 5453 */ 5454 return (-1); 5455 } 5456 } 5457 5458 if (opcode & QL_DBG_OPCODE_RDSTATE) { 5459 /* 5460 * decide which address to use. 5461 */ 5462 if (crbEntry->state_index_a) { 5463 addr = template_hdr->saved_state_array[ 5464 crbEntry-> state_index_a]; 5465 } else { 5466 addr = entry_addr; 5467 } 5468 5469 ret = ql_rdwr_indreg32(ha, addr, &read_value, 1); 5470 if (ret) 5471 return (0); 5472 5473 template_hdr->saved_state_array[crbEntry->state_index_v] 5474 = read_value; 5475 opcode &= ~QL_DBG_OPCODE_RDSTATE; 5476 } 5477 5478 if (opcode & QL_DBG_OPCODE_WRSTATE) { 5479 /* 5480 * decide which value to use. 5481 */ 5482 if (crbEntry->state_index_v) { 5483 read_value = template_hdr->saved_state_array[ 5484 crbEntry->state_index_v]; 5485 } else { 5486 read_value = crbEntry->value_1; 5487 } 5488 /* 5489 * decide which address to use. 5490 */ 5491 if (crbEntry->state_index_a) { 5492 addr = template_hdr->saved_state_array[ 5493 crbEntry-> state_index_a]; 5494 } else { 5495 addr = entry_addr; 5496 } 5497 5498 ret = ql_rdwr_indreg32(ha, addr, &read_value, 0); 5499 if (ret) 5500 return (0); 5501 5502 opcode &= ~QL_DBG_OPCODE_WRSTATE; 5503 } 5504 5505 if (opcode & QL_DBG_OPCODE_MDSTATE) { 5506 /* Read value from saved state using index */ 5507 read_value = template_hdr->saved_state_array[ 5508 crbEntry->state_index_v]; 5509 5510 read_value <<= crbEntry->shl; /*Shift left operation */ 5511 read_value >>= crbEntry->shr; /*Shift right operation */ 5512 5513 if (crbEntry->value_2) { 5514 /* check if AND mask is provided */ 5515 read_value &= crbEntry->value_2; 5516 } 5517 5518 read_value |= crbEntry->value_3; /* OR operation */ 5519 read_value += crbEntry->value_1; /* increment op */ 5520 5521 /* Write value back to state area. */ 5522 5523 template_hdr->saved_state_array[crbEntry->state_index_v] 5524 = read_value; 5525 opcode &= ~QL_DBG_OPCODE_MDSTATE; 5526 } 5527 5528 entry_addr += crbEntry->addr_stride; 5529 } 5530 5531 return (0); 5532 } 5533 5534 /* 5535 * Handling rd poll entry. 5536 */ 5537 5538 static uint32_t 5539 ql_pollrd(qla_host_t *ha, ql_minidump_entry_pollrd_t *entry, 5540 uint32_t *data_buff) 5541 { 5542 int ret; 5543 int loop_cnt; 5544 uint32_t op_count, select_addr, select_value_stride, select_value; 5545 uint32_t read_addr, poll, mask, data_size, data; 5546 uint32_t wait_count = 0; 5547 5548 select_addr = entry->select_addr; 5549 read_addr = entry->read_addr; 5550 select_value = entry->select_value; 5551 select_value_stride = entry->select_value_stride; 5552 op_count = entry->op_count; 5553 poll = entry->poll; 5554 mask = entry->mask; 5555 data_size = entry->data_size; 5556 5557 for (loop_cnt = 0; loop_cnt < op_count; loop_cnt++) { 5558 5559 ret = ql_rdwr_indreg32(ha, select_addr, &select_value, 0); 5560 if (ret) 5561 return (0); 5562 5563 wait_count = 0; 5564 5565 while (wait_count < poll) { 5566 5567 uint32_t temp; 5568 5569 ret = ql_rdwr_indreg32(ha, select_addr, &temp, 1); 5570 if (ret) 5571 return (0); 5572 5573 if ( (temp & mask) != 0 ) { 5574 break; 5575 } 5576 wait_count++; 5577 } 5578 5579 if (wait_count == poll) { 5580 device_printf(ha->pci_dev, 5581 "%s: Error in processing entry\n", __func__); 5582 device_printf(ha->pci_dev, 5583 "%s: wait_count <0x%x> poll <0x%x>\n", 5584 __func__, wait_count, poll); 5585 return 0; 5586 } 5587 5588 ret = ql_rdwr_indreg32(ha, read_addr, &data, 1); 5589 if (ret) 5590 return (0); 5591 5592 *data_buff++ = select_value; 5593 *data_buff++ = data; 5594 select_value = select_value + select_value_stride; 5595 } 5596 5597 /* 5598 * for testing purpose we return amount of data written 5599 */ 5600 return (loop_cnt * (2 * sizeof(uint32_t))); 5601 } 5602 5603 5604 /* 5605 * Handling rd modify write poll entry. 5606 */ 5607 5608 static uint32_t 5609 ql_pollrd_modify_write(qla_host_t *ha, 5610 ql_minidump_entry_rd_modify_wr_with_poll_t *entry, 5611 uint32_t *data_buff) 5612 { 5613 int ret; 5614 uint32_t addr_1, addr_2, value_1, value_2, data; 5615 uint32_t poll, mask, data_size, modify_mask; 5616 uint32_t wait_count = 0; 5617 5618 addr_1 = entry->addr_1; 5619 addr_2 = entry->addr_2; 5620 value_1 = entry->value_1; 5621 value_2 = entry->value_2; 5622 5623 poll = entry->poll; 5624 mask = entry->mask; 5625 modify_mask = entry->modify_mask; 5626 data_size = entry->data_size; 5627 5628 5629 ret = ql_rdwr_indreg32(ha, addr_1, &value_1, 0); 5630 if (ret) 5631 return (0); 5632 5633 wait_count = 0; 5634 while (wait_count < poll) { 5635 5636 uint32_t temp; 5637 5638 ret = ql_rdwr_indreg32(ha, addr_1, &temp, 1); 5639 if (ret) 5640 return (0); 5641 5642 if ( (temp & mask) != 0 ) { 5643 break; 5644 } 5645 wait_count++; 5646 } 5647 5648 if (wait_count == poll) { 5649 device_printf(ha->pci_dev, "%s Error in processing entry\n", 5650 __func__); 5651 } else { 5652 5653 ret = ql_rdwr_indreg32(ha, addr_2, &data, 1); 5654 if (ret) 5655 return (0); 5656 5657 data = (data & modify_mask); 5658 5659 ret = ql_rdwr_indreg32(ha, addr_2, &data, 0); 5660 if (ret) 5661 return (0); 5662 5663 ret = ql_rdwr_indreg32(ha, addr_1, &value_2, 0); 5664 if (ret) 5665 return (0); 5666 5667 /* Poll again */ 5668 wait_count = 0; 5669 while (wait_count < poll) { 5670 5671 uint32_t temp; 5672 5673 ret = ql_rdwr_indreg32(ha, addr_1, &temp, 1); 5674 if (ret) 5675 return (0); 5676 5677 if ( (temp & mask) != 0 ) { 5678 break; 5679 } 5680 wait_count++; 5681 } 5682 *data_buff++ = addr_2; 5683 *data_buff++ = data; 5684 } 5685 5686 /* 5687 * for testing purpose we return amount of data written 5688 */ 5689 return (2 * sizeof(uint32_t)); 5690 } 5691 5692 5693