Linux 5.10-rc1
[linux-block.git] / drivers / misc / mic / scif / scif_main.h
CommitLineData
1802d0be 1/* SPDX-License-Identifier: GPL-2.0-only */
fb4d0e3d
SD
2/*
3 * Intel MIC Platform Software Stack (MPSS)
4 *
5 * Copyright(c) 2014 Intel Corporation.
6 *
fb4d0e3d 7 * Intel SCIF driver.
fb4d0e3d
SD
8 */
9#ifndef SCIF_MAIN_H
10#define SCIF_MAIN_H
11
174cd4b1 12#include <linux/sched/signal.h>
fb4d0e3d
SD
13#include <linux/pci.h>
14#include <linux/miscdevice.h>
15#include <linux/dmaengine.h>
d1824329 16#include <linux/iova.h>
b7f94441 17#include <linux/anon_inodes.h>
fb4d0e3d 18#include <linux/file.h>
d1824329 19#include <linux/vmalloc.h>
fb4d0e3d 20#include <linux/scif.h>
fb4d0e3d
SD
21#include "../common/mic_dev.h"
22
23#define SCIF_MGMT_NODE 0
24#define SCIF_DEFAULT_WATCHDOG_TO 30
25#define SCIF_NODE_ACCEPT_TIMEOUT (3 * HZ)
26#define SCIF_NODE_ALIVE_TIMEOUT (SCIF_DEFAULT_WATCHDOG_TO * HZ)
d1824329 27#define SCIF_RMA_TEMP_CACHE_LIMIT 0x20000
fb4d0e3d
SD
28
29/*
30 * Generic state used for certain node QP message exchanges
31 * like Unregister, Alloc etc.
32 */
33enum scif_msg_state {
34 OP_IDLE = 1,
35 OP_IN_PROGRESS,
36 OP_COMPLETED,
37 OP_FAILED
38};
39
40/*
41 * struct scif_info - Global SCIF information
42 *
43 * @nodeid: Node ID this node is to others
44 * @maxid: Max known node ID
45 * @total: Total number of SCIF nodes
46 * @nr_zombies: number of zombie endpoints
47 * @eplock: Lock to synchronize listening, zombie endpoint lists
48 * @connlock: Lock to synchronize connected and disconnected lists
49 * @nb_connect_lock: Synchronize non blocking connect operations
50 * @port_lock: Synchronize access to SCIF ports
51 * @uaccept: List of user acceptreq waiting for acceptreg
52 * @listen: List of listening end points
53 * @zombie: List of zombie end points with pending RMA's
54 * @connected: List of end points in connected state
55 * @disconnected: List of end points in disconnected state
56 * @nb_connect_list: List for non blocking connections
57 * @misc_work: miscellaneous SCIF tasks
58 * @conflock: Lock to synchronize SCIF node configuration changes
59 * @en_msg_log: Enable debug message logging
60 * @p2p_enable: Enable P2P SCIF network
61 * @mdev: The MISC device
62 * @conn_work: Work for workqueue handling all connections
63 * @exitwq: Wait queue for waiting for an EXIT node QP message response
64 * @loopb_dev: Dummy SCIF device used for loopback
65 * @loopb_wq: Workqueue used for handling loopback messages
66 * @loopb_wqname[16]: Name of loopback workqueue
67 * @loopb_work: Used for submitting work to loopb_wq
68 * @loopb_recv_q: List of messages received on the loopb_wq
69 * @card_initiated_exit: set when the card has initiated the exit
d1824329
SD
70 * @rmalock: Synchronize access to RMA operations
71 * @fencelock: Synchronize access to list of remote fences requested.
72 * @rma: List of temporary registered windows to be destroyed.
73 * @rma_tc: List of temporary registered & cached Windows to be destroyed
74 * @fence: List of remote fence requests
75 * @mmu_notif_work: Work for registration caching MMU notifier workqueue
76 * @mmu_notif_cleanup: List of temporary cached windows for reg cache
77 * @rma_tc_limit: RMA temporary cache limit
fb4d0e3d
SD
78 */
79struct scif_info {
80 u8 nodeid;
81 u8 maxid;
82 u8 total;
83 u32 nr_zombies;
d1824329 84 struct mutex eplock;
fb4d0e3d
SD
85 struct mutex connlock;
86 spinlock_t nb_connect_lock;
87 spinlock_t port_lock;
88 struct list_head uaccept;
89 struct list_head listen;
90 struct list_head zombie;
91 struct list_head connected;
92 struct list_head disconnected;
93 struct list_head nb_connect_list;
94 struct work_struct misc_work;
95 struct mutex conflock;
96 u8 en_msg_log;
97 u8 p2p_enable;
98 struct miscdevice mdev;
99 struct work_struct conn_work;
100 wait_queue_head_t exitwq;
101 struct scif_dev *loopb_dev;
102 struct workqueue_struct *loopb_wq;
103 char loopb_wqname[16];
104 struct work_struct loopb_work;
105 struct list_head loopb_recv_q;
106 bool card_initiated_exit;
d1824329
SD
107 spinlock_t rmalock;
108 struct mutex fencelock;
109 struct list_head rma;
110 struct list_head rma_tc;
111 struct list_head fence;
112 struct work_struct mmu_notif_work;
113 struct list_head mmu_notif_cleanup;
114 unsigned long rma_tc_limit;
fb4d0e3d
SD
115};
116
117/*
118 * struct scif_p2p_info - SCIF mapping information used for P2P
119 *
120 * @ppi_peer_id - SCIF peer node id
121 * @ppi_sg - Scatter list for bar information (One for mmio and one for aper)
122 * @sg_nentries - Number of entries in the scatterlist
123 * @ppi_da: DMA address for MMIO and APER bars
124 * @ppi_len: Length of MMIO and APER bars
125 * @ppi_list: Link in list of mapping information
126 */
127struct scif_p2p_info {
128 u8 ppi_peer_id;
129 struct scatterlist *ppi_sg[2];
130 u64 sg_nentries[2];
131 dma_addr_t ppi_da[2];
132 u64 ppi_len[2];
133#define SCIF_PPI_MMIO 0
134#define SCIF_PPI_APER 1
135 struct list_head ppi_list;
136};
137
138/*
139 * struct scif_dev - SCIF remote device specific fields
140 *
141 * @node: Node id
142 * @p2p: List of P2P mapping information
143 * @qpairs: The node queue pair for exchanging control messages
144 * @intr_wq: Workqueue for handling Node QP messages
145 * @intr_wqname: Name of node QP workqueue for handling interrupts
146 * @intr_bh: Used for submitting work to intr_wq
147 * @lock: Lock used for synchronizing access to the scif device
148 * @sdev: SCIF hardware device on the SCIF hardware bus
149 * @db: doorbell the peer will trigger to generate an interrupt on self
150 * @rdb: Doorbell to trigger on the peer to generate an interrupt on the peer
151 * @cookie: Cookie received while registering the interrupt handler
d3d912eb 152 * @peer_add_work: Work for handling device_add for peer devices
fb4d0e3d
SD
153 * @p2p_dwork: Delayed work to enable polling for P2P state
154 * @qp_dwork: Delayed work for enabling polling for remote QP information
155 * @p2p_retry: Number of times to retry polling of P2P state
156 * @base_addr: P2P aperture bar base address
157 * @mic_mw mmio: The peer MMIO information used for P2P
158 * @spdev: SCIF peer device on the SCIF peer bus
159 * @node_remove_ack_pending: True if a node_remove_ack is pending
160 * @exit_ack_pending: true if an exit_ack is pending
161 * @disconn_wq: Used while waiting for a node remove response
162 * @disconn_rescnt: Keeps track of number of node remove requests sent
163 * @exit: Status of exit message
164 * @qp_dma_addr: Queue pair DMA address passed to the peer
d1824329
SD
165 * @dma_ch_idx: Round robin index for DMA channels
166 * @signal_pool: DMA pool used for scheduling scif_fence_signal DMA's
fb4d0e3d
SD
167*/
168struct scif_dev {
169 u8 node;
170 struct list_head p2p;
171 struct scif_qp *qpairs;
172 struct workqueue_struct *intr_wq;
173 char intr_wqname[16];
174 struct work_struct intr_bh;
175 struct mutex lock;
176 struct scif_hw_dev *sdev;
177 int db;
178 int rdb;
179 struct mic_irq *cookie;
d3d912eb 180 struct work_struct peer_add_work;
fb4d0e3d
SD
181 struct delayed_work p2p_dwork;
182 struct delayed_work qp_dwork;
183 int p2p_retry;
184 dma_addr_t base_addr;
185 struct mic_mw mmio;
186 struct scif_peer_dev __rcu *spdev;
187 bool node_remove_ack_pending;
188 bool exit_ack_pending;
189 wait_queue_head_t disconn_wq;
190 atomic_t disconn_rescnt;
191 enum scif_msg_state exit;
192 dma_addr_t qp_dma_addr;
d1824329
SD
193 int dma_ch_idx;
194 struct dma_pool *signal_pool;
fb4d0e3d
SD
195};
196
d1824329
SD
197extern bool scif_reg_cache_enable;
198extern bool scif_ulimit_check;
fb4d0e3d
SD
199extern struct scif_info scif_info;
200extern struct idr scif_ports;
d3d912eb 201extern struct bus_type scif_peer_bus;
fb4d0e3d
SD
202extern struct scif_dev *scif_dev;
203extern const struct file_operations scif_fops;
b7f94441 204extern const struct file_operations scif_anon_fops;
fb4d0e3d
SD
205
206/* Size of the RB for the Node QP */
207#define SCIF_NODE_QP_SIZE 0x10000
208
209#include "scif_nodeqp.h"
d1824329
SD
210#include "scif_rma.h"
211#include "scif_rma_list.h"
fb4d0e3d
SD
212
213/*
214 * scifdev_self:
215 * @dev: The remote SCIF Device
216 *
217 * Returns true if the SCIF Device passed is the self aka Loopback SCIF device.
218 */
219static inline int scifdev_self(struct scif_dev *dev)
220{
221 return dev->node == scif_info.nodeid;
222}
223
224static inline bool scif_is_mgmt_node(void)
225{
226 return !scif_info.nodeid;
227}
228
229/*
230 * scifdev_is_p2p:
231 * @dev: The remote SCIF Device
232 *
233 * Returns true if the SCIF Device is a MIC Peer to Peer SCIF device.
234 */
235static inline bool scifdev_is_p2p(struct scif_dev *dev)
236{
237 if (scif_is_mgmt_node())
238 return false;
239 else
240 return dev != &scif_dev[SCIF_MGMT_NODE] &&
241 !scifdev_self(dev);
242}
243
244/*
245 * scifdev_alive:
246 * @scifdev: The remote SCIF Device
247 *
248 * Returns true if the remote SCIF Device is running or sleeping for
249 * this endpoint.
250 */
251static inline int _scifdev_alive(struct scif_dev *scifdev)
252{
253 struct scif_peer_dev *spdev;
254
255 rcu_read_lock();
256 spdev = rcu_dereference(scifdev->spdev);
257 rcu_read_unlock();
258 return !!spdev;
259}
260
e9089f43
SD
261#include "scif_epd.h"
262
fb4d0e3d
SD
263void __init scif_init_debugfs(void);
264void scif_exit_debugfs(void);
265int scif_setup_intr_wq(struct scif_dev *scifdev);
266void scif_destroy_intr_wq(struct scif_dev *scifdev);
267void scif_cleanup_scifdev(struct scif_dev *dev);
268void scif_handle_remove_node(int node);
269void scif_disconnect_node(u32 node_id, bool mgmt_initiated);
270void scif_free_qp(struct scif_dev *dev);
271void scif_misc_handler(struct work_struct *work);
272void scif_stop(struct scif_dev *scifdev);
273irqreturn_t scif_intr_handler(int irq, void *data);
274#endif /* SCIF_MAIN_H */