Blame view

drivers/infiniband/hw/hfi1/iowait.h 7.66 KB
772410568   Mike Marciniszyn   IB/hfi1: add driv...
1
2
3
  #ifndef _HFI1_IOWAIT_H
  #define _HFI1_IOWAIT_H
  /*
05d6ac1d8   Jubin John   staging/rdma/hfi1...
4
   * Copyright(c) 2015, 2016 Intel Corporation.
772410568   Mike Marciniszyn   IB/hfi1: add driv...
5
6
7
8
9
10
   *
   * This file is provided under a dual BSD/GPLv2 license.  When using or
   * redistributing this file, you may do so under either license.
   *
   * GPL LICENSE SUMMARY
   *
772410568   Mike Marciniszyn   IB/hfi1: add driv...
11
12
13
14
15
16
17
18
19
20
21
   * This program is free software; you can redistribute it and/or modify
   * it under the terms of version 2 of the GNU General Public License as
   * published by the Free Software Foundation.
   *
   * This program is distributed in the hope that it will be useful, but
   * WITHOUT ANY WARRANTY; without even the implied warranty of
   * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
   * General Public License for more details.
   *
   * BSD LICENSE
   *
772410568   Mike Marciniszyn   IB/hfi1: add driv...
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
   * Redistribution and use in source and binary forms, with or without
   * modification, are permitted provided that the following conditions
   * are met:
   *
   *  - Redistributions of source code must retain the above copyright
   *    notice, this list of conditions and the following disclaimer.
   *  - Redistributions in binary form must reproduce the above copyright
   *    notice, this list of conditions and the following disclaimer in
   *    the documentation and/or other materials provided with the
   *    distribution.
   *  - Neither the name of Intel Corporation nor the names of its
   *    contributors may be used to endorse or promote products derived
   *    from this software without specific prior written permission.
   *
   * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
   * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
   * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
   * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
   * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
   * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
   * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
   * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
   * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
   * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
   * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
   *
   */
  
  #include <linux/list.h>
  #include <linux/workqueue.h>
  #include <linux/sched.h>
711e104dd   Mike Marciniszyn   staging/rdma/hfi1...
53
  #include "sdma_txreq.h"
14553ca11   Mike Marciniszyn   staging/rdma/hfi1...
54

772410568   Mike Marciniszyn   IB/hfi1: add driv...
55
56
57
58
59
60
61
62
63
64
65
66
67
  /*
   * typedef (*restart_t)() - restart callback
   * @work: pointer to work structure
   */
  typedef void (*restart_t)(struct work_struct *work);
  
  struct sdma_txreq;
  struct sdma_engine;
  /**
   * struct iowait - linkage for delayed progress/waiting
   * @list: used to add/insert into QP/PQ wait lists
   * @tx_head: overflow list of sdma_txreq's
   * @sleep: no space callback
a545f5308   Mike Marciniszyn   staging/rdma/hfi:...
68
69
   * @wakeup: space callback wakeup
   * @sdma_drained: sdma count drained
772410568   Mike Marciniszyn   IB/hfi1: add driv...
70
71
   * @iowork: workqueue overhead
   * @wait_dma: wait for sdma_busy == 0
14553ca11   Mike Marciniszyn   staging/rdma/hfi1...
72
   * @wait_pio: wait for pio_busy == 0
772410568   Mike Marciniszyn   IB/hfi1: add driv...
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
   * @sdma_busy: # of packets in flight
   * @count: total number of descriptors in tx_head'ed list
   * @tx_limit: limit for overflow queuing
   * @tx_count: number of tx entry's in tx_head'ed list
   *
   * This is to be embedded in user's state structure
   * (QP or PQ).
   *
   * The sleep and wakeup members are a
   * bit misnamed.   They do not strictly
   * speaking sleep or wake up, but they
   * are callbacks for the ULP to implement
   * what ever queuing/dequeuing of
   * the embedded iowait and its containing struct
   * when a resource shortage like SDMA ring space is seen.
   *
   * Both potentially have locks help
   * so sleeping is not allowed.
   *
   * The wait_dma member along with the iow
   */
  
  struct iowait {
  	struct list_head list;
  	struct list_head tx_head;
  	int (*sleep)(
  		struct sdma_engine *sde,
  		struct iowait *wait,
  		struct sdma_txreq *tx,
  		unsigned seq);
  	void (*wakeup)(struct iowait *wait, int reason);
a545f5308   Mike Marciniszyn   staging/rdma/hfi:...
104
  	void (*sdma_drained)(struct iowait *wait);
772410568   Mike Marciniszyn   IB/hfi1: add driv...
105
106
  	struct work_struct iowork;
  	wait_queue_head_t wait_dma;
14553ca11   Mike Marciniszyn   staging/rdma/hfi1...
107
  	wait_queue_head_t wait_pio;
772410568   Mike Marciniszyn   IB/hfi1: add driv...
108
  	atomic_t sdma_busy;
14553ca11   Mike Marciniszyn   staging/rdma/hfi1...
109
  	atomic_t pio_busy;
772410568   Mike Marciniszyn   IB/hfi1: add driv...
110
111
112
113
114
115
116
117
118
119
120
121
122
  	u32 count;
  	u32 tx_limit;
  	u32 tx_count;
  };
  
  #define SDMA_AVAIL_REASON 0
  
  /**
   * iowait_init() - initialize wait structure
   * @wait: wait struct to initialize
   * @tx_limit: limit for overflow queuing
   * @func: restart function for workqueue
   * @sleep: sleep function for no space
a545f5308   Mike Marciniszyn   staging/rdma/hfi:...
123
   * @resume: wakeup function for no space
772410568   Mike Marciniszyn   IB/hfi1: add driv...
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
   *
   * This function initializes the iowait
   * structure embedded in the QP or PQ.
   *
   */
  
  static inline void iowait_init(
  	struct iowait *wait,
  	u32 tx_limit,
  	void (*func)(struct work_struct *work),
  	int (*sleep)(
  		struct sdma_engine *sde,
  		struct iowait *wait,
  		struct sdma_txreq *tx,
  		unsigned seq),
a545f5308   Mike Marciniszyn   staging/rdma/hfi:...
139
140
  	void (*wakeup)(struct iowait *wait, int reason),
  	void (*sdma_drained)(struct iowait *wait))
772410568   Mike Marciniszyn   IB/hfi1: add driv...
141
142
143
144
145
146
  {
  	wait->count = 0;
  	INIT_LIST_HEAD(&wait->list);
  	INIT_LIST_HEAD(&wait->tx_head);
  	INIT_WORK(&wait->iowork, func);
  	init_waitqueue_head(&wait->wait_dma);
14553ca11   Mike Marciniszyn   staging/rdma/hfi1...
147
  	init_waitqueue_head(&wait->wait_pio);
772410568   Mike Marciniszyn   IB/hfi1: add driv...
148
  	atomic_set(&wait->sdma_busy, 0);
14553ca11   Mike Marciniszyn   staging/rdma/hfi1...
149
  	atomic_set(&wait->pio_busy, 0);
772410568   Mike Marciniszyn   IB/hfi1: add driv...
150
151
152
  	wait->tx_limit = tx_limit;
  	wait->sleep = sleep;
  	wait->wakeup = wakeup;
a545f5308   Mike Marciniszyn   staging/rdma/hfi:...
153
  	wait->sdma_drained = sdma_drained;
772410568   Mike Marciniszyn   IB/hfi1: add driv...
154
155
156
157
158
159
  }
  
  /**
   * iowait_schedule() - initialize wait structure
   * @wait: wait struct to schedule
   * @wq: workqueue for schedule
0a226edd2   Mike Marciniszyn   staging/rdma/hfi1...
160
   * @cpu: cpu
772410568   Mike Marciniszyn   IB/hfi1: add driv...
161
162
163
   */
  static inline void iowait_schedule(
  	struct iowait *wait,
0a226edd2   Mike Marciniszyn   staging/rdma/hfi1...
164
165
  	struct workqueue_struct *wq,
  	int cpu)
772410568   Mike Marciniszyn   IB/hfi1: add driv...
166
  {
0a226edd2   Mike Marciniszyn   staging/rdma/hfi1...
167
  	queue_work_on(cpu, wq, &wait->iowork);
772410568   Mike Marciniszyn   IB/hfi1: add driv...
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
  }
  
  /**
   * iowait_sdma_drain() - wait for DMAs to drain
   *
   * @wait: iowait structure
   *
   * This will delay until the iowait sdmas have
   * completed.
   */
  static inline void iowait_sdma_drain(struct iowait *wait)
  {
  	wait_event(wait->wait_dma, !atomic_read(&wait->sdma_busy));
  }
  
  /**
14553ca11   Mike Marciniszyn   staging/rdma/hfi1...
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
   * iowait_sdma_pending() - return sdma pending count
   *
   * @wait: iowait structure
   *
   */
  static inline int iowait_sdma_pending(struct iowait *wait)
  {
  	return atomic_read(&wait->sdma_busy);
  }
  
  /**
   * iowait_sdma_inc - note sdma io pending
   * @wait: iowait structure
   */
  static inline void iowait_sdma_inc(struct iowait *wait)
  {
  	atomic_inc(&wait->sdma_busy);
  }
  
  /**
   * iowait_sdma_add - add count to pending
   * @wait: iowait structure
   */
  static inline void iowait_sdma_add(struct iowait *wait, int count)
  {
  	atomic_add(count, &wait->sdma_busy);
  }
  
  /**
   * iowait_sdma_dec - note sdma complete
   * @wait: iowait structure
   */
  static inline int iowait_sdma_dec(struct iowait *wait)
  {
  	return atomic_dec_and_test(&wait->sdma_busy);
  }
  
  /**
   * iowait_pio_drain() - wait for pios to drain
   *
   * @wait: iowait structure
   *
   * This will delay until the iowait pios have
   * completed.
   */
  static inline void iowait_pio_drain(struct iowait *wait)
  {
  	wait_event_timeout(wait->wait_pio,
  			   !atomic_read(&wait->pio_busy),
  			   HZ);
  }
  
  /**
   * iowait_pio_pending() - return pio pending count
   *
   * @wait: iowait structure
   *
   */
  static inline int iowait_pio_pending(struct iowait *wait)
  {
  	return atomic_read(&wait->pio_busy);
  }
  
  /**
   * iowait_pio_inc - note pio pending
   * @wait: iowait structure
   */
  static inline void iowait_pio_inc(struct iowait *wait)
  {
  	atomic_inc(&wait->pio_busy);
  }
  
  /**
   * iowait_sdma_dec - note pio complete
   * @wait: iowait structure
   */
  static inline int iowait_pio_dec(struct iowait *wait)
  {
  	return atomic_dec_and_test(&wait->pio_busy);
  }
  
  /**
772410568   Mike Marciniszyn   IB/hfi1: add driv...
266
267
268
269
270
271
272
273
274
   * iowait_drain_wakeup() - trigger iowait_drain() waiter
   *
   * @wait: iowait structure
   *
   * This will trigger any waiters.
   */
  static inline void iowait_drain_wakeup(struct iowait *wait)
  {
  	wake_up(&wait->wait_dma);
14553ca11   Mike Marciniszyn   staging/rdma/hfi1...
275
  	wake_up(&wait->wait_pio);
a545f5308   Mike Marciniszyn   staging/rdma/hfi:...
276
277
  	if (wait->sdma_drained)
  		wait->sdma_drained(wait);
772410568   Mike Marciniszyn   IB/hfi1: add driv...
278
  }
711e104dd   Mike Marciniszyn   staging/rdma/hfi1...
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
  /**
   * iowait_get_txhead() - get packet off of iowait list
   *
   * @wait wait struture
   */
  static inline struct sdma_txreq *iowait_get_txhead(struct iowait *wait)
  {
  	struct sdma_txreq *tx = NULL;
  
  	if (!list_empty(&wait->tx_head)) {
  		tx = list_first_entry(
  			&wait->tx_head,
  			struct sdma_txreq,
  			list);
  		list_del_init(&tx->list);
  	}
  	return tx;
  }
772410568   Mike Marciniszyn   IB/hfi1: add driv...
297
  #endif