]> Git Repo - linux.git/blob - drivers/misc/mic/scif/scif_nm.c
ASoC: simple-card: Use snd_soc_of_parse_aux_devs()
[linux.git] / drivers / misc / mic / scif / scif_nm.c
1 // SPDX-License-Identifier: GPL-2.0-only
2 /*
3  * Intel MIC Platform Software Stack (MPSS)
4  *
5  * Copyright(c) 2014 Intel Corporation.
6  *
7  * Intel SCIF driver.
8  */
9 #include "scif_peer_bus.h"
10
11 #include "scif_main.h"
12 #include "scif_map.h"
13
14 /**
15  * scif_invalidate_ep() - Set state for all connected endpoints
16  * to disconnected and wake up all send/recv waitqueues
17  *
18  * @node: Node to invalidate
19  */
20 static void scif_invalidate_ep(int node)
21 {
22         struct scif_endpt *ep;
23         struct list_head *pos, *tmpq;
24
25         flush_work(&scif_info.conn_work);
26         mutex_lock(&scif_info.connlock);
27         list_for_each_safe(pos, tmpq, &scif_info.disconnected) {
28                 ep = list_entry(pos, struct scif_endpt, list);
29                 if (ep->remote_dev->node == node) {
30                         scif_unmap_all_windows(ep);
31                         spin_lock(&ep->lock);
32                         scif_cleanup_ep_qp(ep);
33                         spin_unlock(&ep->lock);
34                 }
35         }
36         list_for_each_safe(pos, tmpq, &scif_info.connected) {
37                 ep = list_entry(pos, struct scif_endpt, list);
38                 if (ep->remote_dev->node == node) {
39                         list_del(pos);
40                         spin_lock(&ep->lock);
41                         ep->state = SCIFEP_DISCONNECTED;
42                         list_add_tail(&ep->list, &scif_info.disconnected);
43                         scif_cleanup_ep_qp(ep);
44                         wake_up_interruptible(&ep->sendwq);
45                         wake_up_interruptible(&ep->recvwq);
46                         spin_unlock(&ep->lock);
47                         scif_unmap_all_windows(ep);
48                 }
49         }
50         mutex_unlock(&scif_info.connlock);
51 }
52
53 void scif_free_qp(struct scif_dev *scifdev)
54 {
55         struct scif_qp *qp = scifdev->qpairs;
56
57         if (!qp)
58                 return;
59         scif_unmap_single(qp->local_buf, scifdev, qp->inbound_q.size);
60         kfree(qp->inbound_q.rb_base);
61         scif_unmap_single(qp->local_qp, scifdev, sizeof(struct scif_qp));
62         kfree(scifdev->qpairs);
63         scifdev->qpairs = NULL;
64 }
65
66 static void scif_cleanup_qp(struct scif_dev *dev)
67 {
68         struct scif_qp *qp = &dev->qpairs[0];
69
70         if (!qp)
71                 return;
72         scif_iounmap((void *)qp->remote_qp, sizeof(struct scif_qp), dev);
73         scif_iounmap((void *)qp->outbound_q.rb_base,
74                      sizeof(struct scif_qp), dev);
75         qp->remote_qp = NULL;
76         qp->local_write = 0;
77         qp->inbound_q.current_write_offset = 0;
78         qp->inbound_q.current_read_offset = 0;
79         if (scifdev_is_p2p(dev))
80                 scif_free_qp(dev);
81 }
82
83 void scif_send_acks(struct scif_dev *dev)
84 {
85         struct scifmsg msg;
86
87         if (dev->node_remove_ack_pending) {
88                 msg.uop = SCIF_NODE_REMOVE_ACK;
89                 msg.src.node = scif_info.nodeid;
90                 msg.dst.node = SCIF_MGMT_NODE;
91                 msg.payload[0] = dev->node;
92                 scif_nodeqp_send(&scif_dev[SCIF_MGMT_NODE], &msg);
93                 dev->node_remove_ack_pending = false;
94         }
95         if (dev->exit_ack_pending) {
96                 msg.uop = SCIF_EXIT_ACK;
97                 msg.src.node = scif_info.nodeid;
98                 msg.dst.node = dev->node;
99                 scif_nodeqp_send(dev, &msg);
100                 dev->exit_ack_pending = false;
101         }
102 }
103
104 /**
105  * scif_cleanup_scifdev - Uninitialize SCIF data structures for remote
106  *                        SCIF device.
107  * @dev: Remote SCIF device.
108  */
109 void scif_cleanup_scifdev(struct scif_dev *dev)
110 {
111         struct scif_hw_dev *sdev = dev->sdev;
112
113         if (!dev->sdev)
114                 return;
115         if (scifdev_is_p2p(dev)) {
116                 if (dev->cookie) {
117                         sdev->hw_ops->free_irq(sdev, dev->cookie, dev);
118                         dev->cookie = NULL;
119                 }
120                 scif_destroy_intr_wq(dev);
121         }
122         flush_work(&scif_info.misc_work);
123         scif_destroy_p2p(dev);
124         scif_invalidate_ep(dev->node);
125         scif_zap_mmaps(dev->node);
126         scif_cleanup_rma_for_zombies(dev->node);
127         flush_work(&scif_info.misc_work);
128         scif_send_acks(dev);
129         if (!dev->node && scif_info.card_initiated_exit) {
130                 /*
131                  * Send an SCIF_EXIT message which is the last message from MIC
132                  * to the Host and wait for a SCIF_EXIT_ACK
133                  */
134                 scif_send_exit(dev);
135                 scif_info.card_initiated_exit = false;
136         }
137         scif_cleanup_qp(dev);
138 }
139
140 /**
141  * scif_remove_node
142  *
143  * @node: Node to remove
144  */
145 void scif_handle_remove_node(int node)
146 {
147         struct scif_dev *scifdev = &scif_dev[node];
148
149         if (scif_peer_unregister_device(scifdev))
150                 scif_send_acks(scifdev);
151 }
152
153 static int scif_send_rmnode_msg(int node, int remove_node)
154 {
155         struct scifmsg notif_msg;
156         struct scif_dev *dev = &scif_dev[node];
157
158         notif_msg.uop = SCIF_NODE_REMOVE;
159         notif_msg.src.node = scif_info.nodeid;
160         notif_msg.dst.node = node;
161         notif_msg.payload[0] = remove_node;
162         return scif_nodeqp_send(dev, &notif_msg);
163 }
164
165 /**
166  * scif_node_disconnect
167  *
168  * @node_id: source node id [in]
169  * @mgmt_initiated: Disconnection initiated from the mgmt node
170  *
171  * Disconnect a node from the scif network.
172  */
173 void scif_disconnect_node(u32 node_id, bool mgmt_initiated)
174 {
175         int ret;
176         int msg_cnt = 0;
177         u32 i = 0;
178         struct scif_dev *scifdev = &scif_dev[node_id];
179
180         if (!node_id)
181                 return;
182
183         atomic_set(&scifdev->disconn_rescnt, 0);
184
185         /* Destroy p2p network */
186         for (i = 1; i <= scif_info.maxid; i++) {
187                 if (i == node_id)
188                         continue;
189                 ret = scif_send_rmnode_msg(i, node_id);
190                 if (!ret)
191                         msg_cnt++;
192         }
193         /* Wait for the remote nodes to respond with SCIF_NODE_REMOVE_ACK */
194         ret = wait_event_timeout(scifdev->disconn_wq,
195                                  (atomic_read(&scifdev->disconn_rescnt)
196                                  == msg_cnt), SCIF_NODE_ALIVE_TIMEOUT);
197         /* Tell the card to clean up */
198         if (mgmt_initiated && _scifdev_alive(scifdev))
199                 /*
200                  * Send an SCIF_EXIT message which is the last message from Host
201                  * to the MIC and wait for a SCIF_EXIT_ACK
202                  */
203                 scif_send_exit(scifdev);
204         atomic_set(&scifdev->disconn_rescnt, 0);
205         /* Tell the mgmt node to clean up */
206         ret = scif_send_rmnode_msg(SCIF_MGMT_NODE, node_id);
207         if (!ret)
208                 /* Wait for mgmt node to respond with SCIF_NODE_REMOVE_ACK */
209                 wait_event_timeout(scifdev->disconn_wq,
210                                    (atomic_read(&scifdev->disconn_rescnt) == 1),
211                                    SCIF_NODE_ALIVE_TIMEOUT);
212 }
213
214 void scif_get_node_info(void)
215 {
216         struct scifmsg msg;
217         DECLARE_COMPLETION_ONSTACK(node_info);
218
219         msg.uop = SCIF_GET_NODE_INFO;
220         msg.src.node = scif_info.nodeid;
221         msg.dst.node = SCIF_MGMT_NODE;
222         msg.payload[3] = (u64)&node_info;
223
224         if ((scif_nodeqp_send(&scif_dev[SCIF_MGMT_NODE], &msg)))
225                 return;
226
227         /* Wait for a response with SCIF_GET_NODE_INFO */
228         wait_for_completion(&node_info);
229 }
This page took 0.04313 seconds and 4 git commands to generate.