bfa_hw_ct.c 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176
  1. /*
  2. * Copyright (c) 2005-2010 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 "bfad_drv.h"
  18. #include "bfa_modules.h"
  19. #include "bfi_ctreg.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. void __iomem *kva = bfa_ioc_bar0(&bfa->ioc);
  32. if (msix)
  33. writel(vec, kva + __ct_msix_err_vec_reg[fn]);
  34. else
  35. writel(0, kva + __ct_msix_err_vec_reg[fn]);
  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. void __iomem *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_reqq_ack(struct bfa_s *bfa, int reqq)
  78. {
  79. u32 r32;
  80. r32 = readl(bfa->iocfc.bfa_regs.cpe_q_ctrl[reqq]);
  81. writel(r32, bfa->iocfc.bfa_regs.cpe_q_ctrl[reqq]);
  82. }
  83. void
  84. bfa_hwct_rspq_ack(struct bfa_s *bfa, int rspq)
  85. {
  86. u32 r32;
  87. r32 = readl(bfa->iocfc.bfa_regs.rme_q_ctrl[rspq]);
  88. writel(r32, bfa->iocfc.bfa_regs.rme_q_ctrl[rspq]);
  89. }
  90. void
  91. bfa_hwct_msix_getvecs(struct bfa_s *bfa, u32 *msix_vecs_bmap,
  92. u32 *num_vecs, u32 *max_vec_bit)
  93. {
  94. *msix_vecs_bmap = (1 << BFA_MSIX_CT_MAX) - 1;
  95. *max_vec_bit = (1 << (BFA_MSIX_CT_MAX - 1));
  96. *num_vecs = BFA_MSIX_CT_MAX;
  97. }
  98. /*
  99. * Setup MSI-X vector for catapult
  100. */
  101. void
  102. bfa_hwct_msix_init(struct bfa_s *bfa, int nvecs)
  103. {
  104. WARN_ON((nvecs != 1) && (nvecs != BFA_MSIX_CT_MAX));
  105. bfa_trc(bfa, nvecs);
  106. bfa->msix.nvecs = nvecs;
  107. bfa_hwct_msix_uninstall(bfa);
  108. }
  109. void
  110. bfa_hwct_msix_install(struct bfa_s *bfa)
  111. {
  112. int i;
  113. if (bfa->msix.nvecs == 0)
  114. return;
  115. if (bfa->msix.nvecs == 1) {
  116. for (i = 0; i < BFA_MSIX_CT_MAX; i++)
  117. bfa->msix.handler[i] = bfa_msix_all;
  118. return;
  119. }
  120. for (i = BFA_MSIX_CPE_Q0; i <= BFA_MSIX_CPE_Q3; i++)
  121. bfa->msix.handler[i] = bfa_msix_reqq;
  122. for (; i <= BFA_MSIX_RME_Q3; i++)
  123. bfa->msix.handler[i] = bfa_msix_rspq;
  124. WARN_ON(i != BFA_MSIX_LPU_ERR);
  125. bfa->msix.handler[BFA_MSIX_LPU_ERR] = bfa_msix_lpu_err;
  126. }
  127. void
  128. bfa_hwct_msix_uninstall(struct bfa_s *bfa)
  129. {
  130. int i;
  131. for (i = 0; i < BFA_MSIX_CT_MAX; i++)
  132. bfa->msix.handler[i] = bfa_hwct_msix_dummy;
  133. }
  134. /*
  135. * Enable MSI-X vectors
  136. */
  137. void
  138. bfa_hwct_isr_mode_set(struct bfa_s *bfa, bfa_boolean_t msix)
  139. {
  140. bfa_trc(bfa, 0);
  141. bfa_hwct_msix_lpu_err_set(bfa, msix, BFA_MSIX_LPU_ERR);
  142. bfa_ioc_isr_mode_set(&bfa->ioc, msix);
  143. }
  144. void
  145. bfa_hwct_msix_get_rme_range(struct bfa_s *bfa, u32 *start, u32 *end)
  146. {
  147. *start = BFA_MSIX_RME_Q0;
  148. *end = BFA_MSIX_RME_Q3;
  149. }