2 * Copyright (c) 2009-2010 Chelsio, Inc. All rights reserved.
4 * This software is available to you under a choice of one of two
5 * licenses. You may choose to be licensed under the terms of the GNU
6 * General Public License (GPL) Version 2, available from the file
7 * COPYING in the main directory of this source tree, or the
8 * OpenIB.org BSD license below:
10 * Redistribution and use in source and binary forms, with or
11 * without modification, are permitted provided that the following
14 * - Redistributions of source code must retain the above
15 * copyright notice, this list of conditions and the following
18 * - Redistributions in binary form must reproduce the above
19 * copyright notice, this list of conditions and the following
20 * disclaimer in the documentation and/or other materials
21 * provided with the distribution.
23 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
24 * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
25 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
26 * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
27 * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
28 * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
29 * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
32 /* Crude resource management */
33 #include <linux/kernel.h>
34 #include <linux/random.h>
35 #include <linux/slab.h>
36 #include <linux/kfifo.h>
37 #include <linux/spinlock.h>
38 #include <linux/errno.h>
39 #include <linux/genalloc.h>
40 #include <linux/ratelimit.h>
43 #define RANDOM_SIZE 16
45 static int __c4iw_init_resource_fifo(struct kfifo
*fifo
,
46 spinlock_t
*fifo_lock
,
51 u32 i
, j
, entry
= 0, idx
;
54 spin_lock_init(fifo_lock
);
56 if (kfifo_alloc(fifo
, nr
* sizeof(u32
), GFP_KERNEL
))
59 for (i
= 0; i
< skip_low
+ skip_high
; i
++)
60 kfifo_in(fifo
, (unsigned char *) &entry
, sizeof(u32
));
63 random_bytes
= random32();
64 for (i
= 0; i
< RANDOM_SIZE
; i
++)
65 rarray
[i
] = i
+ skip_low
;
66 for (i
= skip_low
+ RANDOM_SIZE
; i
< nr
- skip_high
; i
++) {
67 if (j
>= RANDOM_SIZE
) {
69 random_bytes
= random32();
71 idx
= (random_bytes
>> (j
* 2)) & 0xF;
73 (unsigned char *) &rarray
[idx
],
78 for (i
= 0; i
< RANDOM_SIZE
; i
++)
80 (unsigned char *) &rarray
[i
],
83 for (i
= skip_low
; i
< nr
- skip_high
; i
++)
84 kfifo_in(fifo
, (unsigned char *) &i
, sizeof(u32
));
86 for (i
= 0; i
< skip_low
+ skip_high
; i
++)
87 if (kfifo_out_locked(fifo
, (unsigned char *) &entry
,
88 sizeof(u32
), fifo_lock
))
93 static int c4iw_init_resource_fifo(struct kfifo
*fifo
, spinlock_t
* fifo_lock
,
94 u32 nr
, u32 skip_low
, u32 skip_high
)
96 return __c4iw_init_resource_fifo(fifo
, fifo_lock
, nr
, skip_low
,
100 static int c4iw_init_resource_fifo_random(struct kfifo
*fifo
,
101 spinlock_t
*fifo_lock
,
102 u32 nr
, u32 skip_low
, u32 skip_high
)
104 return __c4iw_init_resource_fifo(fifo
, fifo_lock
, nr
, skip_low
,
108 static int c4iw_init_qid_fifo(struct c4iw_rdev
*rdev
)
112 spin_lock_init(&rdev
->resource
.qid_fifo_lock
);
114 if (kfifo_alloc(&rdev
->resource
.qid_fifo
, rdev
->lldi
.vr
->qp
.size
*
115 sizeof(u32
), GFP_KERNEL
))
118 for (i
= rdev
->lldi
.vr
->qp
.start
;
119 i
< rdev
->lldi
.vr
->qp
.start
+ rdev
->lldi
.vr
->qp
.size
; i
++)
120 if (!(i
& rdev
->qpmask
))
121 kfifo_in(&rdev
->resource
.qid_fifo
,
122 (unsigned char *) &i
, sizeof(u32
));
126 /* nr_* must be power of 2 */
127 int c4iw_init_resource(struct c4iw_rdev
*rdev
, u32 nr_tpt
, u32 nr_pdid
)
130 err
= c4iw_init_resource_fifo_random(&rdev
->resource
.tpt_fifo
,
131 &rdev
->resource
.tpt_fifo_lock
,
135 err
= c4iw_init_qid_fifo(rdev
);
138 err
= c4iw_init_resource_fifo(&rdev
->resource
.pdid_fifo
,
139 &rdev
->resource
.pdid_fifo_lock
,
145 kfifo_free(&rdev
->resource
.qid_fifo
);
147 kfifo_free(&rdev
->resource
.tpt_fifo
);
153 * returns 0 if no resource available
155 u32
c4iw_get_resource(struct kfifo
*fifo
, spinlock_t
*lock
)
158 if (kfifo_out_locked(fifo
, (unsigned char *) &entry
, sizeof(u32
), lock
))
164 void c4iw_put_resource(struct kfifo
*fifo
, u32 entry
, spinlock_t
*lock
)
166 PDBG("%s entry 0x%x\n", __func__
, entry
);
167 kfifo_in_locked(fifo
, (unsigned char *) &entry
, sizeof(u32
), lock
);
170 u32
c4iw_get_cqid(struct c4iw_rdev
*rdev
, struct c4iw_dev_ucontext
*uctx
)
172 struct c4iw_qid_list
*entry
;
176 mutex_lock(&uctx
->lock
);
177 if (!list_empty(&uctx
->cqids
)) {
178 entry
= list_entry(uctx
->cqids
.next
, struct c4iw_qid_list
,
180 list_del(&entry
->entry
);
184 qid
= c4iw_get_resource(&rdev
->resource
.qid_fifo
,
185 &rdev
->resource
.qid_fifo_lock
);
188 for (i
= qid
+1; i
& rdev
->qpmask
; i
++) {
189 entry
= kmalloc(sizeof *entry
, GFP_KERNEL
);
193 list_add_tail(&entry
->entry
, &uctx
->cqids
);
197 * now put the same ids on the qp list since they all
198 * map to the same db/gts page.
200 entry
= kmalloc(sizeof *entry
, GFP_KERNEL
);
204 list_add_tail(&entry
->entry
, &uctx
->qpids
);
205 for (i
= qid
+1; i
& rdev
->qpmask
; i
++) {
206 entry
= kmalloc(sizeof *entry
, GFP_KERNEL
);
210 list_add_tail(&entry
->entry
, &uctx
->qpids
);
214 mutex_unlock(&uctx
->lock
);
215 PDBG("%s qid 0x%x\n", __func__
, qid
);
219 void c4iw_put_cqid(struct c4iw_rdev
*rdev
, u32 qid
,
220 struct c4iw_dev_ucontext
*uctx
)
222 struct c4iw_qid_list
*entry
;
224 entry
= kmalloc(sizeof *entry
, GFP_KERNEL
);
227 PDBG("%s qid 0x%x\n", __func__
, qid
);
229 mutex_lock(&uctx
->lock
);
230 list_add_tail(&entry
->entry
, &uctx
->cqids
);
231 mutex_unlock(&uctx
->lock
);
234 u32
c4iw_get_qpid(struct c4iw_rdev
*rdev
, struct c4iw_dev_ucontext
*uctx
)
236 struct c4iw_qid_list
*entry
;
240 mutex_lock(&uctx
->lock
);
241 if (!list_empty(&uctx
->qpids
)) {
242 entry
= list_entry(uctx
->qpids
.next
, struct c4iw_qid_list
,
244 list_del(&entry
->entry
);
248 qid
= c4iw_get_resource(&rdev
->resource
.qid_fifo
,
249 &rdev
->resource
.qid_fifo_lock
);
252 for (i
= qid
+1; i
& rdev
->qpmask
; i
++) {
253 entry
= kmalloc(sizeof *entry
, GFP_KERNEL
);
257 list_add_tail(&entry
->entry
, &uctx
->qpids
);
261 * now put the same ids on the cq list since they all
262 * map to the same db/gts page.
264 entry
= kmalloc(sizeof *entry
, GFP_KERNEL
);
268 list_add_tail(&entry
->entry
, &uctx
->cqids
);
269 for (i
= qid
; i
& rdev
->qpmask
; i
++) {
270 entry
= kmalloc(sizeof *entry
, GFP_KERNEL
);
274 list_add_tail(&entry
->entry
, &uctx
->cqids
);
278 mutex_unlock(&uctx
->lock
);
279 PDBG("%s qid 0x%x\n", __func__
, qid
);
283 void c4iw_put_qpid(struct c4iw_rdev
*rdev
, u32 qid
,
284 struct c4iw_dev_ucontext
*uctx
)
286 struct c4iw_qid_list
*entry
;
288 entry
= kmalloc(sizeof *entry
, GFP_KERNEL
);
291 PDBG("%s qid 0x%x\n", __func__
, qid
);
293 mutex_lock(&uctx
->lock
);
294 list_add_tail(&entry
->entry
, &uctx
->qpids
);
295 mutex_unlock(&uctx
->lock
);
298 void c4iw_destroy_resource(struct c4iw_resource
*rscp
)
300 kfifo_free(&rscp
->tpt_fifo
);
301 kfifo_free(&rscp
->qid_fifo
);
302 kfifo_free(&rscp
->pdid_fifo
);
306 * PBL Memory Manager. Uses Linux generic allocator.
309 #define MIN_PBL_SHIFT 8 /* 256B == min PBL size (32 entries) */
311 u32
c4iw_pblpool_alloc(struct c4iw_rdev
*rdev
, int size
)
313 unsigned long addr
= gen_pool_alloc(rdev
->pbl_pool
, size
);
314 PDBG("%s addr 0x%x size %d\n", __func__
, (u32
)addr
, size
);
316 printk_ratelimited(KERN_WARNING MOD
"%s: Out of PBL memory\n",
317 pci_name(rdev
->lldi
.pdev
));
321 void c4iw_pblpool_free(struct c4iw_rdev
*rdev
, u32 addr
, int size
)
323 PDBG("%s addr 0x%x size %d\n", __func__
, addr
, size
);
324 gen_pool_free(rdev
->pbl_pool
, (unsigned long)addr
, size
);
327 int c4iw_pblpool_create(struct c4iw_rdev
*rdev
)
329 unsigned pbl_start
, pbl_chunk
, pbl_top
;
331 rdev
->pbl_pool
= gen_pool_create(MIN_PBL_SHIFT
, -1);
335 pbl_start
= rdev
->lldi
.vr
->pbl
.start
;
336 pbl_chunk
= rdev
->lldi
.vr
->pbl
.size
;
337 pbl_top
= pbl_start
+ pbl_chunk
;
339 while (pbl_start
< pbl_top
) {
340 pbl_chunk
= min(pbl_top
- pbl_start
+ 1, pbl_chunk
);
341 if (gen_pool_add(rdev
->pbl_pool
, pbl_start
, pbl_chunk
, -1)) {
342 PDBG("%s failed to add PBL chunk (%x/%x)\n",
343 __func__
, pbl_start
, pbl_chunk
);
344 if (pbl_chunk
<= 1024 << MIN_PBL_SHIFT
) {
345 printk(KERN_WARNING MOD
346 "Failed to add all PBL chunks (%x/%x)\n",
348 pbl_top
- pbl_start
);
353 PDBG("%s added PBL chunk (%x/%x)\n",
354 __func__
, pbl_start
, pbl_chunk
);
355 pbl_start
+= pbl_chunk
;
362 void c4iw_pblpool_destroy(struct c4iw_rdev
*rdev
)
364 gen_pool_destroy(rdev
->pbl_pool
);
368 * RQT Memory Manager. Uses Linux generic allocator.
371 #define MIN_RQT_SHIFT 10 /* 1KB == min RQT size (16 entries) */
373 u32
c4iw_rqtpool_alloc(struct c4iw_rdev
*rdev
, int size
)
375 unsigned long addr
= gen_pool_alloc(rdev
->rqt_pool
, size
<< 6);
376 PDBG("%s addr 0x%x size %d\n", __func__
, (u32
)addr
, size
<< 6);
378 printk_ratelimited(KERN_WARNING MOD
"%s: Out of RQT memory\n",
379 pci_name(rdev
->lldi
.pdev
));
383 void c4iw_rqtpool_free(struct c4iw_rdev
*rdev
, u32 addr
, int size
)
385 PDBG("%s addr 0x%x size %d\n", __func__
, addr
, size
<< 6);
386 gen_pool_free(rdev
->rqt_pool
, (unsigned long)addr
, size
<< 6);
389 int c4iw_rqtpool_create(struct c4iw_rdev
*rdev
)
391 unsigned rqt_start
, rqt_chunk
, rqt_top
;
393 rdev
->rqt_pool
= gen_pool_create(MIN_RQT_SHIFT
, -1);
397 rqt_start
= rdev
->lldi
.vr
->rq
.start
;
398 rqt_chunk
= rdev
->lldi
.vr
->rq
.size
;
399 rqt_top
= rqt_start
+ rqt_chunk
;
401 while (rqt_start
< rqt_top
) {
402 rqt_chunk
= min(rqt_top
- rqt_start
+ 1, rqt_chunk
);
403 if (gen_pool_add(rdev
->rqt_pool
, rqt_start
, rqt_chunk
, -1)) {
404 PDBG("%s failed to add RQT chunk (%x/%x)\n",
405 __func__
, rqt_start
, rqt_chunk
);
406 if (rqt_chunk
<= 1024 << MIN_RQT_SHIFT
) {
407 printk(KERN_WARNING MOD
408 "Failed to add all RQT chunks (%x/%x)\n",
409 rqt_start
, rqt_top
- rqt_start
);
414 PDBG("%s added RQT chunk (%x/%x)\n",
415 __func__
, rqt_start
, rqt_chunk
);
416 rqt_start
+= rqt_chunk
;
422 void c4iw_rqtpool_destroy(struct c4iw_rdev
*rdev
)
424 gen_pool_destroy(rdev
->rqt_pool
);
430 #define MIN_OCQP_SHIFT 12 /* 4KB == min ocqp size */
432 u32
c4iw_ocqp_pool_alloc(struct c4iw_rdev
*rdev
, int size
)
434 unsigned long addr
= gen_pool_alloc(rdev
->ocqp_pool
, size
);
435 PDBG("%s addr 0x%x size %d\n", __func__
, (u32
)addr
, size
);
439 void c4iw_ocqp_pool_free(struct c4iw_rdev
*rdev
, u32 addr
, int size
)
441 PDBG("%s addr 0x%x size %d\n", __func__
, addr
, size
);
442 gen_pool_free(rdev
->ocqp_pool
, (unsigned long)addr
, size
);
445 int c4iw_ocqp_pool_create(struct c4iw_rdev
*rdev
)
447 unsigned start
, chunk
, top
;
449 rdev
->ocqp_pool
= gen_pool_create(MIN_OCQP_SHIFT
, -1);
450 if (!rdev
->ocqp_pool
)
453 start
= rdev
->lldi
.vr
->ocq
.start
;
454 chunk
= rdev
->lldi
.vr
->ocq
.size
;
457 while (start
< top
) {
458 chunk
= min(top
- start
+ 1, chunk
);
459 if (gen_pool_add(rdev
->ocqp_pool
, start
, chunk
, -1)) {
460 PDBG("%s failed to add OCQP chunk (%x/%x)\n",
461 __func__
, start
, chunk
);
462 if (chunk
<= 1024 << MIN_OCQP_SHIFT
) {
463 printk(KERN_WARNING MOD
464 "Failed to add all OCQP chunks (%x/%x)\n",
470 PDBG("%s added OCQP chunk (%x/%x)\n",
471 __func__
, start
, chunk
);
478 void c4iw_ocqp_pool_destroy(struct c4iw_rdev
*rdev
)
480 gen_pool_destroy(rdev
->ocqp_pool
);