if_pci.h 6.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245
  1. /*
  2. * Copyright (c) 2013-2016 The Linux Foundation. All rights reserved.
  3. *
  4. * Previously licensed under the ISC license by Qualcomm Atheros, Inc.
  5. *
  6. *
  7. * Permission to use, copy, modify, and/or distribute this software for
  8. * any purpose with or without fee is hereby granted, provided that the
  9. * above copyright notice and this permission notice appear in all
  10. * copies.
  11. *
  12. * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL
  13. * WARRANTIES WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED
  14. * WARRANTIES OF MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE
  15. * AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL
  16. * DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR
  17. * PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
  18. * TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
  19. * PERFORMANCE OF THIS SOFTWARE.
  20. */
  21. /*
  22. * This file was originally distributed by Qualcomm Atheros, Inc.
  23. * under proprietary terms before Copyright ownership was assigned
  24. * to the Linux Foundation.
  25. */
  26. #ifndef __ATH_PCI_H__
  27. #define __ATH_PCI_H__
  28. #include <linux/version.h>
  29. #include <linux/semaphore.h>
  30. #include <linux/interrupt.h>
  31. #define ATH_DBG_DEFAULT 0
  32. #include <ol_if_athvar.h>
  33. #include "hif.h"
  34. #include "cepci.h"
  35. #include "ce_main.h"
  36. /* An address (e.g. of a buffer) in Copy Engine space. */
  37. #define HIF_MAX_TASKLET_NUM 11
  38. struct hif_tasklet_entry {
  39. uint8_t id; /* 0 - 9: maps to CE, 10: fw */
  40. void *hif_handler; /* struct hif_pci_softc */
  41. };
  42. /**
  43. * enum hif_pm_runtime_state - Driver States for Runtime Power Management
  44. * HIF_PM_RUNTIME_STATE_NONE: runtime pm is off
  45. * HIF_PM_RUNTIME_STATE_ON: runtime pm is active and link is active
  46. * HIF_PM_RUNTIME_STATE_INPROGRESS: a runtime suspend or resume is in progress
  47. * HIF_PM_RUNTIME_STATE_SUSPENDED: the driver is runtime suspended
  48. */
  49. enum hif_pm_runtime_state {
  50. HIF_PM_RUNTIME_STATE_NONE,
  51. HIF_PM_RUNTIME_STATE_ON,
  52. HIF_PM_RUNTIME_STATE_INPROGRESS,
  53. HIF_PM_RUNTIME_STATE_SUSPENDED,
  54. };
  55. #ifdef FEATURE_RUNTIME_PM
  56. /**
  57. * struct hif_pm_runtime_lock - data structure for preventing runtime suspend
  58. * @list - global list of runtime locks
  59. * @active - true if this lock is preventing suspend
  60. * @name - character string for tracking this lock
  61. */
  62. struct hif_pm_runtime_lock {
  63. struct list_head list;
  64. bool active;
  65. uint32_t timeout;
  66. const char *name;
  67. };
  68. /* Debugging stats for Runtime PM */
  69. struct hif_pci_pm_stats {
  70. u32 suspended;
  71. u32 suspend_err;
  72. u32 resumed;
  73. u32 runtime_get;
  74. u32 runtime_put;
  75. u32 request_resume;
  76. u32 allow_suspend;
  77. u32 prevent_suspend;
  78. u32 prevent_suspend_timeout;
  79. u32 allow_suspend_timeout;
  80. u32 runtime_get_err;
  81. void *last_resume_caller;
  82. unsigned long suspend_jiffies;
  83. };
  84. #endif
  85. /**
  86. * struct hif_msi_info - Structure to hold msi info
  87. * @magic: cookie
  88. * @magic_da: dma address
  89. * @dmaContext: dma address
  90. *
  91. * Structure to hold MSI information for PCIe interrupts
  92. */
  93. struct hif_msi_info {
  94. void *magic;
  95. dma_addr_t magic_da;
  96. OS_DMA_MEM_CONTEXT(dmacontext)
  97. };
  98. struct hif_pci_softc {
  99. struct HIF_CE_state ce_sc;
  100. void __iomem *mem; /* PCI address. */
  101. /* For efficiency, should be first in struct */
  102. struct device *dev;
  103. struct pci_dev *pdev;
  104. int num_msi_intrs; /* number of MSI interrupts granted */
  105. /* 0 --> using legacy PCI line interrupts */
  106. struct tasklet_struct intr_tq; /* tasklet */
  107. struct hif_msi_info msi_info;
  108. int irq;
  109. int irq_event;
  110. int cacheline_sz;
  111. u16 devid;
  112. qdf_dma_addr_t soc_pcie_bar0;
  113. struct hif_tasklet_entry tasklet_entries[HIF_MAX_TASKLET_NUM];
  114. bool pci_enabled;
  115. qdf_spinlock_t irq_lock;
  116. qdf_work_t reschedule_tasklet_work;
  117. uint32_t lcr_val;
  118. #ifdef FEATURE_RUNTIME_PM
  119. atomic_t pm_state;
  120. uint32_t prevent_suspend_cnt;
  121. struct hif_pci_pm_stats pm_stats;
  122. struct work_struct pm_work;
  123. spinlock_t runtime_lock;
  124. struct timer_list runtime_timer;
  125. struct list_head prevent_suspend_list;
  126. unsigned long runtime_timer_expires;
  127. struct hif_pm_runtime_lock *prevent_linkdown_lock;
  128. #ifdef WLAN_OPEN_SOURCE
  129. struct dentry *pm_dentry;
  130. #endif
  131. #endif
  132. };
  133. bool hif_pci_targ_is_present(struct hif_softc *scn, void *__iomem *mem);
  134. int hif_configure_irq(struct hif_softc *sc);
  135. void hif_pci_cancel_deferred_target_sleep(struct hif_softc *scn);
  136. /*
  137. * A firmware interrupt to the Host is indicated by the
  138. * low bit of SCRATCH_3_ADDRESS being set.
  139. */
  140. #define FW_EVENT_PENDING_REG_ADDRESS SCRATCH_3_ADDRESS
  141. /*
  142. * Typically, MSI Interrupts are used with PCIe. To force use of legacy
  143. * "ABCD" PCI line interrupts rather than MSI, define
  144. * FORCE_LEGACY_PCI_INTERRUPTS.
  145. * Even when NOT forced, the driver may attempt to use legacy PCI interrupts
  146. * MSI allocation fails
  147. */
  148. #define LEGACY_INTERRUPTS(sc) ((sc)->num_msi_intrs == 0)
  149. /*
  150. * There may be some pending tx frames during platform suspend.
  151. * Suspend operation should be delayed until those tx frames are
  152. * transfered from the host to target. This macro specifies how
  153. * long suspend thread has to sleep before checking pending tx
  154. * frame count.
  155. */
  156. #define OL_ATH_TX_DRAIN_WAIT_DELAY 50 /* ms */
  157. #define HIF_CE_DRAIN_WAIT_DELAY 10 /* ms */
  158. /*
  159. * Wait time (in unit of OL_ATH_TX_DRAIN_WAIT_DELAY) for pending
  160. * tx frame completion before suspend. Refer: hif_pci_suspend()
  161. */
  162. #ifndef QCA_WIFI_3_0_EMU
  163. #define OL_ATH_TX_DRAIN_WAIT_CNT 10
  164. #else
  165. #define OL_ATH_TX_DRAIN_WAIT_CNT 60
  166. #endif
  167. #define HIF_CE_DRAIN_WAIT_CNT 20
  168. #ifdef FEATURE_RUNTIME_PM
  169. #include <linux/pm_runtime.h>
  170. #ifdef WLAN_OPEN_SOURCE
  171. static inline int hif_pm_request_resume(struct device *dev)
  172. {
  173. return pm_request_resume(dev);
  174. }
  175. static inline int __hif_pm_runtime_get(struct device *dev)
  176. {
  177. return pm_runtime_get(dev);
  178. }
  179. static inline int hif_pm_runtime_put_auto(struct device *dev)
  180. {
  181. return pm_runtime_put_autosuspend(dev);
  182. }
  183. static inline void hif_pm_runtime_mark_last_busy(struct device *dev)
  184. {
  185. pm_runtime_mark_last_busy(dev);
  186. }
  187. static inline int hif_pm_runtime_resume(struct device *dev)
  188. {
  189. return pm_runtime_resume(dev);
  190. }
  191. #else
  192. static inline int hif_pm_request_resume(struct device *dev)
  193. {
  194. return cnss_pm_runtime_request(dev, CNSS_PM_REQUEST_RESUME);
  195. }
  196. static inline int __hif_pm_runtime_get(struct device *dev)
  197. {
  198. return cnss_pm_runtime_request(dev, CNSS_PM_RUNTIME_GET);
  199. }
  200. static inline int hif_pm_runtime_put_auto(struct device *dev)
  201. {
  202. return cnss_pm_runtime_request(dev, CNSS_PM_RUNTIME_PUT_AUTO);
  203. }
  204. static inline void hif_pm_runtime_mark_last_busy(struct device *dev)
  205. {
  206. cnss_pm_runtime_request(dev, CNSS_PM_RUNTIME_MARK_LAST_BUSY);
  207. }
  208. static inline int hif_pm_runtime_resume(struct device *dev)
  209. {
  210. return cnss_pm_runtime_request(dev, CNSS_PM_RUNTIME_RESUME);
  211. }
  212. #endif /* WLAN_OPEN_SOURCE */
  213. #else
  214. static inline void hif_pm_runtime_mark_last_busy(struct device *dev) { }
  215. #endif /* FEATURE_RUNTIME_PM */
  216. #endif /* __ATH_PCI_H__ */