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