bfa_hw_ct.c 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162
  1. /*
  2. * Copyright (c) 2005-2009 Brocade Communications Systems, Inc.
  3. * All rights reserved
  4. * www.brocade.com
  5. *
  6. * Linux driver for Brocade Fibre Channel Host Bus Adapter.
  7. *
  8. * This program is free software; you can redistribute it and/or modify it
  9. * under the terms of the GNU General Public License (GPL) Version 2 as
  10. * published by the Free Software Foundation
  11. *
  12. * This program is distributed in the hope that it will be useful, but
  13. * WITHOUT ANY WARRANTY; without even the implied warranty of
  14. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  15. * General Public License for more details.
  16. */
  17. #include <bfa_priv.h>
  18. #include <bfi/bfi_ctreg.h>
  19. #include <bfa_ioc.h>
  20. BFA_TRC_FILE(HAL, IOCFC_CT);
  21. static u32 __ct_msix_err_vec_reg[] = {
  22. HOST_MSIX_ERR_INDEX_FN0,
  23. HOST_MSIX_ERR_INDEX_FN1,
  24. HOST_MSIX_ERR_INDEX_FN2,
  25. HOST_MSIX_ERR_INDEX_FN3,
  26. };
  27. static void
  28. bfa_hwct_msix_lpu_err_set(struct bfa_s *bfa, bfa_boolean_t msix, int vec)
  29. {
  30. int fn = bfa_ioc_pcifn(&bfa->ioc);
  31. bfa_os_addr_t kva = bfa_ioc_bar0(&bfa->ioc);
  32. if (msix)
  33. bfa_reg_write(kva + __ct_msix_err_vec_reg[fn], vec);
  34. else
  35. bfa_reg_write(kva + __ct_msix_err_vec_reg[fn], 0);
  36. }
  37. /**
  38. * Dummy interrupt handler for handling spurious interrupt during chip-reinit.
  39. */
  40. static void
  41. bfa_hwct_msix_dummy(struct bfa_s *bfa, int vec)
  42. {
  43. }
  44. void
  45. bfa_hwct_reginit(struct bfa_s *bfa)
  46. {
  47. struct bfa_iocfc_regs_s *bfa_regs = &bfa->iocfc.bfa_regs;
  48. bfa_os_addr_t kva = bfa_ioc_bar0(&bfa->ioc);
  49. int i, q, fn = bfa_ioc_pcifn(&bfa->ioc);
  50. if (fn == 0) {
  51. bfa_regs->intr_status = (kva + HOSTFN0_INT_STATUS);
  52. bfa_regs->intr_mask = (kva + HOSTFN0_INT_MSK);
  53. } else {
  54. bfa_regs->intr_status = (kva + HOSTFN1_INT_STATUS);
  55. bfa_regs->intr_mask = (kva + HOSTFN1_INT_MSK);
  56. }
  57. for (i = 0; i < BFI_IOC_MAX_CQS; i++) {
  58. /*
  59. * CPE registers
  60. */
  61. q = CPE_Q_NUM(fn, i);
  62. bfa_regs->cpe_q_pi[i] = (kva + CPE_PI_PTR_Q(q << 5));
  63. bfa_regs->cpe_q_ci[i] = (kva + CPE_CI_PTR_Q(q << 5));
  64. bfa_regs->cpe_q_depth[i] = (kva + CPE_DEPTH_Q(q << 5));
  65. bfa_regs->cpe_q_ctrl[i] = (kva + CPE_QCTRL_Q(q << 5));
  66. /*
  67. * RME registers
  68. */
  69. q = CPE_Q_NUM(fn, i);
  70. bfa_regs->rme_q_pi[i] = (kva + RME_PI_PTR_Q(q << 5));
  71. bfa_regs->rme_q_ci[i] = (kva + RME_CI_PTR_Q(q << 5));
  72. bfa_regs->rme_q_depth[i] = (kva + RME_DEPTH_Q(q << 5));
  73. bfa_regs->rme_q_ctrl[i] = (kva + RME_QCTRL_Q(q << 5));
  74. }
  75. }
  76. void
  77. bfa_hwct_rspq_ack(struct bfa_s *bfa, int rspq)
  78. {
  79. u32 r32;
  80. r32 = bfa_reg_read(bfa->iocfc.bfa_regs.rme_q_ctrl[rspq]);
  81. bfa_reg_write(bfa->iocfc.bfa_regs.rme_q_ctrl[rspq], r32);
  82. }
  83. void
  84. bfa_hwct_msix_getvecs(struct bfa_s *bfa, u32 *msix_vecs_bmap,
  85. u32 *num_vecs, u32 *max_vec_bit)
  86. {
  87. *msix_vecs_bmap = (1 << BFA_MSIX_CT_MAX) - 1;
  88. *max_vec_bit = (1 << (BFA_MSIX_CT_MAX - 1));
  89. *num_vecs = BFA_MSIX_CT_MAX;
  90. }
  91. /**
  92. * Setup MSI-X vector for catapult
  93. */
  94. void
  95. bfa_hwct_msix_init(struct bfa_s *bfa, int nvecs)
  96. {
  97. bfa_assert((nvecs == 1) || (nvecs == BFA_MSIX_CT_MAX));
  98. bfa_trc(bfa, nvecs);
  99. bfa->msix.nvecs = nvecs;
  100. bfa_hwct_msix_uninstall(bfa);
  101. }
  102. void
  103. bfa_hwct_msix_install(struct bfa_s *bfa)
  104. {
  105. int i;
  106. if (bfa->msix.nvecs == 0)
  107. return;
  108. if (bfa->msix.nvecs == 1) {
  109. for (i = 0; i < BFA_MSIX_CT_MAX; i++)
  110. bfa->msix.handler[i] = bfa_msix_all;
  111. return;
  112. }
  113. for (i = BFA_MSIX_CPE_Q0; i <= BFA_MSIX_CPE_Q3; i++)
  114. bfa->msix.handler[i] = bfa_msix_reqq;
  115. for (; i <= BFA_MSIX_RME_Q3; i++)
  116. bfa->msix.handler[i] = bfa_msix_rspq;
  117. bfa_assert(i == BFA_MSIX_LPU_ERR);
  118. bfa->msix.handler[BFA_MSIX_LPU_ERR] = bfa_msix_lpu_err;
  119. }
  120. void
  121. bfa_hwct_msix_uninstall(struct bfa_s *bfa)
  122. {
  123. int i;
  124. for (i = 0; i < BFA_MSIX_CT_MAX; i++)
  125. bfa->msix.handler[i] = bfa_hwct_msix_dummy;
  126. }
  127. /**
  128. * Enable MSI-X vectors
  129. */
  130. void
  131. bfa_hwct_isr_mode_set(struct bfa_s *bfa, bfa_boolean_t msix)
  132. {
  133. bfa_trc(bfa, 0);
  134. bfa_hwct_msix_lpu_err_set(bfa, msix, BFA_MSIX_LPU_ERR);
  135. bfa_ioc_isr_mode_set(&bfa->ioc, msix);
  136. }