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