sas_event.c 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216
  1. // SPDX-License-Identifier: GPL-2.0
  2. /*
  3. * Serial Attached SCSI (SAS) Event processing
  4. *
  5. * Copyright (C) 2005 Adaptec, Inc. All rights reserved.
  6. * Copyright (C) 2005 Luben Tuikov <[email protected]>
  7. */
  8. #include <linux/export.h>
  9. #include <scsi/scsi_host.h>
  10. #include "sas_internal.h"
  11. bool sas_queue_work(struct sas_ha_struct *ha, struct sas_work *sw)
  12. {
  13. if (!test_bit(SAS_HA_REGISTERED, &ha->state))
  14. return false;
  15. if (test_bit(SAS_HA_DRAINING, &ha->state)) {
  16. /* add it to the defer list, if not already pending */
  17. if (list_empty(&sw->drain_node))
  18. list_add_tail(&sw->drain_node, &ha->defer_q);
  19. return true;
  20. }
  21. return queue_work(ha->event_q, &sw->work);
  22. }
  23. static bool sas_queue_event(int event, struct sas_work *work,
  24. struct sas_ha_struct *ha)
  25. {
  26. unsigned long flags;
  27. bool rc;
  28. spin_lock_irqsave(&ha->lock, flags);
  29. rc = sas_queue_work(ha, work);
  30. spin_unlock_irqrestore(&ha->lock, flags);
  31. return rc;
  32. }
  33. void sas_queue_deferred_work(struct sas_ha_struct *ha)
  34. {
  35. struct sas_work *sw, *_sw;
  36. spin_lock_irq(&ha->lock);
  37. list_for_each_entry_safe(sw, _sw, &ha->defer_q, drain_node) {
  38. list_del_init(&sw->drain_node);
  39. if (!sas_queue_work(ha, sw)) {
  40. pm_runtime_put(ha->dev);
  41. sas_free_event(to_asd_sas_event(&sw->work));
  42. }
  43. }
  44. spin_unlock_irq(&ha->lock);
  45. }
  46. void __sas_drain_work(struct sas_ha_struct *ha)
  47. {
  48. set_bit(SAS_HA_DRAINING, &ha->state);
  49. /* flush submitters */
  50. spin_lock_irq(&ha->lock);
  51. spin_unlock_irq(&ha->lock);
  52. drain_workqueue(ha->event_q);
  53. drain_workqueue(ha->disco_q);
  54. clear_bit(SAS_HA_DRAINING, &ha->state);
  55. sas_queue_deferred_work(ha);
  56. }
  57. int sas_drain_work(struct sas_ha_struct *ha)
  58. {
  59. int err;
  60. err = mutex_lock_interruptible(&ha->drain_mutex);
  61. if (err)
  62. return err;
  63. if (test_bit(SAS_HA_REGISTERED, &ha->state))
  64. __sas_drain_work(ha);
  65. mutex_unlock(&ha->drain_mutex);
  66. return 0;
  67. }
  68. EXPORT_SYMBOL_GPL(sas_drain_work);
  69. void sas_disable_revalidation(struct sas_ha_struct *ha)
  70. {
  71. mutex_lock(&ha->disco_mutex);
  72. set_bit(SAS_HA_ATA_EH_ACTIVE, &ha->state);
  73. mutex_unlock(&ha->disco_mutex);
  74. }
  75. void sas_enable_revalidation(struct sas_ha_struct *ha)
  76. {
  77. int i;
  78. mutex_lock(&ha->disco_mutex);
  79. clear_bit(SAS_HA_ATA_EH_ACTIVE, &ha->state);
  80. for (i = 0; i < ha->num_phys; i++) {
  81. struct asd_sas_port *port = ha->sas_port[i];
  82. const int ev = DISCE_REVALIDATE_DOMAIN;
  83. struct sas_discovery *d = &port->disc;
  84. struct asd_sas_phy *sas_phy;
  85. if (!test_and_clear_bit(ev, &d->pending))
  86. continue;
  87. spin_lock(&port->phy_list_lock);
  88. if (list_empty(&port->phy_list)) {
  89. spin_unlock(&port->phy_list_lock);
  90. continue;
  91. }
  92. sas_phy = container_of(port->phy_list.next, struct asd_sas_phy,
  93. port_phy_el);
  94. spin_unlock(&port->phy_list_lock);
  95. sas_notify_port_event(sas_phy,
  96. PORTE_BROADCAST_RCVD, GFP_KERNEL);
  97. }
  98. mutex_unlock(&ha->disco_mutex);
  99. }
  100. static void sas_port_event_worker(struct work_struct *work)
  101. {
  102. struct asd_sas_event *ev = to_asd_sas_event(work);
  103. struct asd_sas_phy *phy = ev->phy;
  104. struct sas_ha_struct *ha = phy->ha;
  105. sas_port_event_fns[ev->event](work);
  106. pm_runtime_put(ha->dev);
  107. sas_free_event(ev);
  108. }
  109. static void sas_phy_event_worker(struct work_struct *work)
  110. {
  111. struct asd_sas_event *ev = to_asd_sas_event(work);
  112. struct asd_sas_phy *phy = ev->phy;
  113. struct sas_ha_struct *ha = phy->ha;
  114. sas_phy_event_fns[ev->event](work);
  115. pm_runtime_put(ha->dev);
  116. sas_free_event(ev);
  117. }
  118. /* defer works of new phys during suspend */
  119. static bool sas_defer_event(struct asd_sas_phy *phy, struct asd_sas_event *ev)
  120. {
  121. struct sas_ha_struct *ha = phy->ha;
  122. unsigned long flags;
  123. bool deferred = false;
  124. spin_lock_irqsave(&ha->lock, flags);
  125. if (test_bit(SAS_HA_RESUMING, &ha->state) && !phy->suspended) {
  126. struct sas_work *sw = &ev->work;
  127. list_add_tail(&sw->drain_node, &ha->defer_q);
  128. deferred = true;
  129. }
  130. spin_unlock_irqrestore(&ha->lock, flags);
  131. return deferred;
  132. }
  133. void sas_notify_port_event(struct asd_sas_phy *phy, enum port_event event,
  134. gfp_t gfp_flags)
  135. {
  136. struct sas_ha_struct *ha = phy->ha;
  137. struct asd_sas_event *ev;
  138. BUG_ON(event >= PORT_NUM_EVENTS);
  139. ev = sas_alloc_event(phy, gfp_flags);
  140. if (!ev)
  141. return;
  142. /* Call pm_runtime_put() with pairs in sas_port_event_worker() */
  143. pm_runtime_get_noresume(ha->dev);
  144. INIT_SAS_EVENT(ev, sas_port_event_worker, phy, event);
  145. if (sas_defer_event(phy, ev))
  146. return;
  147. if (!sas_queue_event(event, &ev->work, ha)) {
  148. pm_runtime_put(ha->dev);
  149. sas_free_event(ev);
  150. }
  151. }
  152. EXPORT_SYMBOL_GPL(sas_notify_port_event);
  153. void sas_notify_phy_event(struct asd_sas_phy *phy, enum phy_event event,
  154. gfp_t gfp_flags)
  155. {
  156. struct sas_ha_struct *ha = phy->ha;
  157. struct asd_sas_event *ev;
  158. BUG_ON(event >= PHY_NUM_EVENTS);
  159. ev = sas_alloc_event(phy, gfp_flags);
  160. if (!ev)
  161. return;
  162. /* Call pm_runtime_put() with pairs in sas_phy_event_worker() */
  163. pm_runtime_get_noresume(ha->dev);
  164. INIT_SAS_EVENT(ev, sas_phy_event_worker, phy, event);
  165. if (sas_defer_event(phy, ev))
  166. return;
  167. if (!sas_queue_event(event, &ev->work, ha)) {
  168. pm_runtime_put(ha->dev);
  169. sas_free_event(ev);
  170. }
  171. }
  172. EXPORT_SYMBOL_GPL(sas_notify_phy_event);