blob: f7256d88d38f99499aacb82cfb28ebdf23b8a9bd [file] [log] [blame]
oulijun9a443532016-07-21 19:06:38 +08001/*
2 * Copyright (c) 2016 Hisilicon Limited.
3 * Copyright (c) 2007, 2008 Mellanox Technologies. All rights reserved.
4 *
5 * This software is available to you under a choice of one of two
6 * licenses. You may choose to be licensed under the terms of the GNU
7 * General Public License (GPL) Version 2, available from the file
8 * COPYING in the main directory of this source tree, or the
9 * OpenIB.org BSD license below:
10 *
11 * Redistribution and use in source and binary forms, with or
12 * without modification, are permitted provided that the following
13 * conditions are met:
14 *
15 * - Redistributions of source code must retain the above
16 * copyright notice, this list of conditions and the following
17 * disclaimer.
18 *
19 * - Redistributions in binary form must reproduce the above
20 * copyright notice, this list of conditions and the following
21 * disclaimer in the documentation and/or other materials
22 * provided with the distribution.
23 *
24 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
25 * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
26 * MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
27 * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS
28 * BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN
29 * ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
30 * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
31 * SOFTWARE.
32 */
33
34#include <linux/platform_device.h>
Matan Barake89bf462017-06-08 17:23:50 +030035#include <linux/vmalloc.h>
oulijun9a443532016-07-21 19:06:38 +080036#include <rdma/ib_umem.h>
37#include "hns_roce_device.h"
38#include "hns_roce_cmd.h"
39#include "hns_roce_hem.h"
40
41static u32 hw_index_to_key(unsigned long ind)
42{
43 return (u32)(ind >> 24) | (ind << 8);
44}
45
Shaobo Xubfcc6812016-11-29 23:10:26 +000046unsigned long key_to_hw_index(u32 key)
oulijun9a443532016-07-21 19:06:38 +080047{
48 return (key << 24) | (key >> 8);
49}
Wei Hu(Xavier)08805fd2017-08-30 17:22:59 +080050EXPORT_SYMBOL_GPL(key_to_hw_index);
oulijun9a443532016-07-21 19:06:38 +080051
52static int hns_roce_sw2hw_mpt(struct hns_roce_dev *hr_dev,
53 struct hns_roce_cmd_mailbox *mailbox,
54 unsigned long mpt_index)
55{
56 return hns_roce_cmd_mbox(hr_dev, mailbox->dma, 0, mpt_index, 0,
57 HNS_ROCE_CMD_SW2HW_MPT,
Wei Hu (Xavier)6b877c32016-11-23 19:41:05 +000058 HNS_ROCE_CMD_TIMEOUT_MSECS);
oulijun9a443532016-07-21 19:06:38 +080059}
60
Shaobo Xubfcc6812016-11-29 23:10:26 +000061int hns_roce_hw2sw_mpt(struct hns_roce_dev *hr_dev,
oulijun9a443532016-07-21 19:06:38 +080062 struct hns_roce_cmd_mailbox *mailbox,
63 unsigned long mpt_index)
64{
65 return hns_roce_cmd_mbox(hr_dev, 0, mailbox ? mailbox->dma : 0,
66 mpt_index, !mailbox, HNS_ROCE_CMD_HW2SW_MPT,
Wei Hu (Xavier)6b877c32016-11-23 19:41:05 +000067 HNS_ROCE_CMD_TIMEOUT_MSECS);
oulijun9a443532016-07-21 19:06:38 +080068}
Wei Hu(Xavier)08805fd2017-08-30 17:22:59 +080069EXPORT_SYMBOL_GPL(hns_roce_hw2sw_mpt);
oulijun9a443532016-07-21 19:06:38 +080070
71static int hns_roce_buddy_alloc(struct hns_roce_buddy *buddy, int order,
72 unsigned long *seg)
73{
74 int o;
75 u32 m;
76
77 spin_lock(&buddy->lock);
78
79 for (o = order; o <= buddy->max_order; ++o) {
80 if (buddy->num_free[o]) {
81 m = 1 << (buddy->max_order - o);
82 *seg = find_first_bit(buddy->bits[o], m);
83 if (*seg < m)
84 goto found;
85 }
86 }
87 spin_unlock(&buddy->lock);
88 return -1;
89
90 found:
91 clear_bit(*seg, buddy->bits[o]);
92 --buddy->num_free[o];
93
94 while (o > order) {
95 --o;
96 *seg <<= 1;
97 set_bit(*seg ^ 1, buddy->bits[o]);
98 ++buddy->num_free[o];
99 }
100
101 spin_unlock(&buddy->lock);
102
103 *seg <<= order;
104 return 0;
105}
106
107static void hns_roce_buddy_free(struct hns_roce_buddy *buddy, unsigned long seg,
108 int order)
109{
110 seg >>= order;
111
112 spin_lock(&buddy->lock);
113
114 while (test_bit(seg ^ 1, buddy->bits[order])) {
115 clear_bit(seg ^ 1, buddy->bits[order]);
116 --buddy->num_free[order];
117 seg >>= 1;
118 ++order;
119 }
120
121 set_bit(seg, buddy->bits[order]);
122 ++buddy->num_free[order];
123
124 spin_unlock(&buddy->lock);
125}
126
127static int hns_roce_buddy_init(struct hns_roce_buddy *buddy, int max_order)
128{
129 int i, s;
130
131 buddy->max_order = max_order;
132 spin_lock_init(&buddy->lock);
Markus Elfring4418b272017-02-16 09:30:55 +0100133 buddy->bits = kcalloc(buddy->max_order + 1,
134 sizeof(*buddy->bits),
135 GFP_KERNEL);
136 buddy->num_free = kcalloc(buddy->max_order + 1,
137 sizeof(*buddy->num_free),
138 GFP_KERNEL);
oulijun9a443532016-07-21 19:06:38 +0800139 if (!buddy->bits || !buddy->num_free)
140 goto err_out;
141
142 for (i = 0; i <= buddy->max_order; ++i) {
143 s = BITS_TO_LONGS(1 << (buddy->max_order - i));
Wei Hu (Xavier)8d497eb2016-11-23 19:41:01 +0000144 buddy->bits[i] = kcalloc(s, sizeof(long), GFP_KERNEL |
145 __GFP_NOWARN);
146 if (!buddy->bits[i]) {
147 buddy->bits[i] = vzalloc(s * sizeof(long));
148 if (!buddy->bits[i])
149 goto err_out_free;
150 }
oulijun9a443532016-07-21 19:06:38 +0800151 }
152
153 set_bit(0, buddy->bits[buddy->max_order]);
154 buddy->num_free[buddy->max_order] = 1;
155
156 return 0;
157
158err_out_free:
159 for (i = 0; i <= buddy->max_order; ++i)
Wei Hu (Xavier)8d497eb2016-11-23 19:41:01 +0000160 kvfree(buddy->bits[i]);
oulijun9a443532016-07-21 19:06:38 +0800161
162err_out:
163 kfree(buddy->bits);
164 kfree(buddy->num_free);
165 return -ENOMEM;
166}
167
168static void hns_roce_buddy_cleanup(struct hns_roce_buddy *buddy)
169{
170 int i;
171
172 for (i = 0; i <= buddy->max_order; ++i)
Wei Hu (Xavier)8d497eb2016-11-23 19:41:01 +0000173 kvfree(buddy->bits[i]);
oulijun9a443532016-07-21 19:06:38 +0800174
175 kfree(buddy->bits);
176 kfree(buddy->num_free);
177}
178
179static int hns_roce_alloc_mtt_range(struct hns_roce_dev *hr_dev, int order,
Shaobo Xu9766edc2017-08-30 17:23:09 +0800180 unsigned long *seg, u32 mtt_type)
oulijun9a443532016-07-21 19:06:38 +0800181{
182 struct hns_roce_mr_table *mr_table = &hr_dev->mr_table;
Shaobo Xu9766edc2017-08-30 17:23:09 +0800183 struct hns_roce_hem_table *table;
184 struct hns_roce_buddy *buddy;
185 int ret;
oulijun9a443532016-07-21 19:06:38 +0800186
Shaobo Xu9766edc2017-08-30 17:23:09 +0800187 if (mtt_type == MTT_TYPE_WQE) {
188 buddy = &mr_table->mtt_buddy;
189 table = &mr_table->mtt_table;
190 } else {
191 buddy = &mr_table->mtt_cqe_buddy;
192 table = &mr_table->mtt_cqe_table;
193 }
194
195 ret = hns_roce_buddy_alloc(buddy, order, seg);
oulijun9a443532016-07-21 19:06:38 +0800196 if (ret == -1)
197 return -1;
198
Shaobo Xu9766edc2017-08-30 17:23:09 +0800199 if (hns_roce_table_get_range(hr_dev, table, *seg,
oulijun9a443532016-07-21 19:06:38 +0800200 *seg + (1 << order) - 1)) {
Shaobo Xu9766edc2017-08-30 17:23:09 +0800201 hns_roce_buddy_free(buddy, *seg, order);
oulijun9a443532016-07-21 19:06:38 +0800202 return -1;
203 }
204
205 return 0;
206}
207
208int hns_roce_mtt_init(struct hns_roce_dev *hr_dev, int npages, int page_shift,
209 struct hns_roce_mtt *mtt)
210{
Shaobo Xu9766edc2017-08-30 17:23:09 +0800211 int ret;
oulijun9a443532016-07-21 19:06:38 +0800212 int i;
213
214 /* Page num is zero, correspond to DMA memory register */
215 if (!npages) {
216 mtt->order = -1;
217 mtt->page_shift = HNS_ROCE_HEM_PAGE_SHIFT;
218 return 0;
219 }
220
Stephen Boydad61dd32017-05-08 15:57:50 -0700221 /* Note: if page_shift is zero, FAST memory register */
oulijun9a443532016-07-21 19:06:38 +0800222 mtt->page_shift = page_shift;
223
224 /* Compute MTT entry necessary */
225 for (mtt->order = 0, i = HNS_ROCE_MTT_ENTRY_PER_SEG; i < npages;
226 i <<= 1)
227 ++mtt->order;
228
229 /* Allocate MTT entry */
Shaobo Xu9766edc2017-08-30 17:23:09 +0800230 ret = hns_roce_alloc_mtt_range(hr_dev, mtt->order, &mtt->first_seg,
231 mtt->mtt_type);
oulijun9a443532016-07-21 19:06:38 +0800232 if (ret == -1)
233 return -ENOMEM;
234
235 return 0;
236}
237
238void hns_roce_mtt_cleanup(struct hns_roce_dev *hr_dev, struct hns_roce_mtt *mtt)
239{
240 struct hns_roce_mr_table *mr_table = &hr_dev->mr_table;
241
242 if (mtt->order < 0)
243 return;
244
Shaobo Xu9766edc2017-08-30 17:23:09 +0800245 if (mtt->mtt_type == MTT_TYPE_WQE) {
246 hns_roce_buddy_free(&mr_table->mtt_buddy, mtt->first_seg,
247 mtt->order);
248 hns_roce_table_put_range(hr_dev, &mr_table->mtt_table,
249 mtt->first_seg,
250 mtt->first_seg + (1 << mtt->order) - 1);
251 } else {
252 hns_roce_buddy_free(&mr_table->mtt_cqe_buddy, mtt->first_seg,
253 mtt->order);
254 hns_roce_table_put_range(hr_dev, &mr_table->mtt_cqe_table,
255 mtt->first_seg,
256 mtt->first_seg + (1 << mtt->order) - 1);
257 }
oulijun9a443532016-07-21 19:06:38 +0800258}
Wei Hu(Xavier)08805fd2017-08-30 17:22:59 +0800259EXPORT_SYMBOL_GPL(hns_roce_mtt_cleanup);
oulijun9a443532016-07-21 19:06:38 +0800260
Wei Hu(Xavier)ff795f72017-08-30 17:23:10 +0800261static void hns_roce_loop_free(struct hns_roce_dev *hr_dev,
262 struct hns_roce_mr *mr, int err_loop_index,
263 int loop_i, int loop_j)
264{
265 struct device *dev = hr_dev->dev;
266 u32 mhop_num;
267 u32 pbl_bt_sz;
268 u64 bt_idx;
269 int i, j;
270
271 pbl_bt_sz = 1 << (hr_dev->caps.pbl_ba_pg_sz + PAGE_SHIFT);
272 mhop_num = hr_dev->caps.pbl_hop_num;
273
274 i = loop_i;
Wei Hu(Xavier)ff795f72017-08-30 17:23:10 +0800275 if (mhop_num == 3 && err_loop_index == 2) {
276 for (; i >= 0; i--) {
277 dma_free_coherent(dev, pbl_bt_sz, mr->pbl_bt_l1[i],
278 mr->pbl_l1_dma_addr[i]);
279
280 for (j = 0; j < pbl_bt_sz / 8; j++) {
281 if (i == loop_i && j >= loop_j)
282 break;
283
284 bt_idx = i * pbl_bt_sz / 8 + j;
285 dma_free_coherent(dev, pbl_bt_sz,
286 mr->pbl_bt_l2[bt_idx],
287 mr->pbl_l2_dma_addr[bt_idx]);
288 }
289 }
290 } else if (mhop_num == 3 && err_loop_index == 1) {
291 for (i -= 1; i >= 0; i--) {
292 dma_free_coherent(dev, pbl_bt_sz, mr->pbl_bt_l1[i],
293 mr->pbl_l1_dma_addr[i]);
294
295 for (j = 0; j < pbl_bt_sz / 8; j++) {
296 bt_idx = i * pbl_bt_sz / 8 + j;
297 dma_free_coherent(dev, pbl_bt_sz,
298 mr->pbl_bt_l2[bt_idx],
299 mr->pbl_l2_dma_addr[bt_idx]);
300 }
301 }
302 } else if (mhop_num == 2 && err_loop_index == 1) {
303 for (i -= 1; i >= 0; i--)
304 dma_free_coherent(dev, pbl_bt_sz, mr->pbl_bt_l1[i],
305 mr->pbl_l1_dma_addr[i]);
306 } else {
307 dev_warn(dev, "not support: mhop_num=%d, err_loop_index=%d.",
308 mhop_num, err_loop_index);
309 return;
310 }
311
312 dma_free_coherent(dev, pbl_bt_sz, mr->pbl_bt_l0, mr->pbl_l0_dma_addr);
313 mr->pbl_bt_l0 = NULL;
314 mr->pbl_l0_dma_addr = 0;
315}
316
317/* PBL multi hop addressing */
318static int hns_roce_mhop_alloc(struct hns_roce_dev *hr_dev, int npages,
319 struct hns_roce_mr *mr)
320{
321 struct device *dev = hr_dev->dev;
322 int mr_alloc_done = 0;
323 int npages_allocated;
324 int i = 0, j = 0;
325 u32 pbl_bt_sz;
326 u32 mhop_num;
327 u64 pbl_last_bt_num;
328 u64 pbl_bt_cnt = 0;
329 u64 bt_idx;
330 u64 size;
331
332 mhop_num = hr_dev->caps.pbl_hop_num;
333 pbl_bt_sz = 1 << (hr_dev->caps.pbl_ba_pg_sz + PAGE_SHIFT);
334 pbl_last_bt_num = (npages + pbl_bt_sz / 8 - 1) / (pbl_bt_sz / 8);
335
336 if (mhop_num == HNS_ROCE_HOP_NUM_0)
337 return 0;
338
339 /* hop_num = 1 */
340 if (mhop_num == 1) {
341 if (npages > pbl_bt_sz / 8) {
342 dev_err(dev, "npages %d is larger than buf_pg_sz!",
343 npages);
344 return -EINVAL;
345 }
346 mr->pbl_buf = dma_alloc_coherent(dev, npages * 8,
347 &(mr->pbl_dma_addr),
348 GFP_KERNEL);
349 if (!mr->pbl_buf)
350 return -ENOMEM;
351
352 mr->pbl_size = npages;
353 mr->pbl_ba = mr->pbl_dma_addr;
354 mr->pbl_hop_num = hr_dev->caps.pbl_hop_num;
355 mr->pbl_ba_pg_sz = hr_dev->caps.pbl_ba_pg_sz;
356 mr->pbl_buf_pg_sz = hr_dev->caps.pbl_buf_pg_sz;
357 return 0;
358 }
359
360 mr->pbl_l1_dma_addr = kcalloc(pbl_bt_sz / 8,
361 sizeof(*mr->pbl_l1_dma_addr),
362 GFP_KERNEL);
363 if (!mr->pbl_l1_dma_addr)
364 return -ENOMEM;
365
366 mr->pbl_bt_l1 = kcalloc(pbl_bt_sz / 8, sizeof(*mr->pbl_bt_l1),
367 GFP_KERNEL);
368 if (!mr->pbl_bt_l1)
369 goto err_kcalloc_bt_l1;
370
371 if (mhop_num == 3) {
372 mr->pbl_l2_dma_addr = kcalloc(pbl_last_bt_num,
373 sizeof(*mr->pbl_l2_dma_addr),
374 GFP_KERNEL);
375 if (!mr->pbl_l2_dma_addr)
376 goto err_kcalloc_l2_dma;
377
378 mr->pbl_bt_l2 = kcalloc(pbl_last_bt_num,
379 sizeof(*mr->pbl_bt_l2),
380 GFP_KERNEL);
381 if (!mr->pbl_bt_l2)
382 goto err_kcalloc_bt_l2;
383 }
384
385 /* alloc L0 BT */
386 mr->pbl_bt_l0 = dma_alloc_coherent(dev, pbl_bt_sz,
387 &(mr->pbl_l0_dma_addr),
388 GFP_KERNEL);
389 if (!mr->pbl_bt_l0)
390 goto err_dma_alloc_l0;
391
392 if (mhop_num == 2) {
393 /* alloc L1 BT */
394 for (i = 0; i < pbl_bt_sz / 8; i++) {
395 if (pbl_bt_cnt + 1 < pbl_last_bt_num) {
396 size = pbl_bt_sz;
397 } else {
398 npages_allocated = i * (pbl_bt_sz / 8);
399 size = (npages - npages_allocated) * 8;
400 }
401 mr->pbl_bt_l1[i] = dma_alloc_coherent(dev, size,
402 &(mr->pbl_l1_dma_addr[i]),
403 GFP_KERNEL);
404 if (!mr->pbl_bt_l1[i]) {
405 hns_roce_loop_free(hr_dev, mr, 1, i, 0);
406 goto err_dma_alloc_l0;
407 }
408
409 *(mr->pbl_bt_l0 + i) = mr->pbl_l1_dma_addr[i];
410
411 pbl_bt_cnt++;
412 if (pbl_bt_cnt >= pbl_last_bt_num)
413 break;
414 }
415 } else if (mhop_num == 3) {
416 /* alloc L1, L2 BT */
417 for (i = 0; i < pbl_bt_sz / 8; i++) {
418 mr->pbl_bt_l1[i] = dma_alloc_coherent(dev, pbl_bt_sz,
419 &(mr->pbl_l1_dma_addr[i]),
420 GFP_KERNEL);
421 if (!mr->pbl_bt_l1[i]) {
422 hns_roce_loop_free(hr_dev, mr, 1, i, 0);
423 goto err_dma_alloc_l0;
424 }
425
426 *(mr->pbl_bt_l0 + i) = mr->pbl_l1_dma_addr[i];
427
428 for (j = 0; j < pbl_bt_sz / 8; j++) {
429 bt_idx = i * pbl_bt_sz / 8 + j;
430
431 if (pbl_bt_cnt + 1 < pbl_last_bt_num) {
432 size = pbl_bt_sz;
433 } else {
434 npages_allocated = bt_idx *
435 (pbl_bt_sz / 8);
436 size = (npages - npages_allocated) * 8;
437 }
438 mr->pbl_bt_l2[bt_idx] = dma_alloc_coherent(
439 dev, size,
440 &(mr->pbl_l2_dma_addr[bt_idx]),
441 GFP_KERNEL);
442 if (!mr->pbl_bt_l2[bt_idx]) {
443 hns_roce_loop_free(hr_dev, mr, 2, i, j);
444 goto err_dma_alloc_l0;
445 }
446
447 *(mr->pbl_bt_l1[i] + j) =
448 mr->pbl_l2_dma_addr[bt_idx];
449
450 pbl_bt_cnt++;
451 if (pbl_bt_cnt >= pbl_last_bt_num) {
452 mr_alloc_done = 1;
453 break;
454 }
455 }
456
457 if (mr_alloc_done)
458 break;
459 }
460 }
461
462 mr->l0_chunk_last_num = i + 1;
463 if (mhop_num == 3)
464 mr->l1_chunk_last_num = j + 1;
465
466 mr->pbl_size = npages;
467 mr->pbl_ba = mr->pbl_l0_dma_addr;
468 mr->pbl_hop_num = hr_dev->caps.pbl_hop_num;
469 mr->pbl_ba_pg_sz = hr_dev->caps.pbl_ba_pg_sz;
470 mr->pbl_buf_pg_sz = hr_dev->caps.pbl_buf_pg_sz;
471
472 return 0;
473
474err_dma_alloc_l0:
475 kfree(mr->pbl_bt_l2);
476 mr->pbl_bt_l2 = NULL;
477
478err_kcalloc_bt_l2:
479 kfree(mr->pbl_l2_dma_addr);
480 mr->pbl_l2_dma_addr = NULL;
481
482err_kcalloc_l2_dma:
483 kfree(mr->pbl_bt_l1);
484 mr->pbl_bt_l1 = NULL;
485
486err_kcalloc_bt_l1:
487 kfree(mr->pbl_l1_dma_addr);
488 mr->pbl_l1_dma_addr = NULL;
489
490 return -ENOMEM;
491}
492
oulijun9a443532016-07-21 19:06:38 +0800493static int hns_roce_mr_alloc(struct hns_roce_dev *hr_dev, u32 pd, u64 iova,
494 u64 size, u32 access, int npages,
495 struct hns_roce_mr *mr)
496{
Wei Hu(Xavier)13ca9702017-08-30 17:23:02 +0800497 struct device *dev = hr_dev->dev;
oulijun9a443532016-07-21 19:06:38 +0800498 unsigned long index = 0;
499 int ret = 0;
oulijun9a443532016-07-21 19:06:38 +0800500
501 /* Allocate a key for mr from mr_table */
502 ret = hns_roce_bitmap_alloc(&hr_dev->mr_table.mtpt_bitmap, &index);
503 if (ret == -1)
504 return -ENOMEM;
505
506 mr->iova = iova; /* MR va starting addr */
507 mr->size = size; /* MR addr range */
508 mr->pd = pd; /* MR num */
509 mr->access = access; /* MR access permit */
510 mr->enabled = 0; /* MR active status */
511 mr->key = hw_index_to_key(index); /* MR key */
512
513 if (size == ~0ull) {
514 mr->type = MR_TYPE_DMA;
515 mr->pbl_buf = NULL;
516 mr->pbl_dma_addr = 0;
Wei Hu(Xavier)ff795f72017-08-30 17:23:10 +0800517 /* PBL multi-hop addressing parameters */
518 mr->pbl_bt_l2 = NULL;
519 mr->pbl_bt_l1 = NULL;
520 mr->pbl_bt_l0 = NULL;
521 mr->pbl_l2_dma_addr = NULL;
522 mr->pbl_l1_dma_addr = NULL;
523 mr->pbl_l0_dma_addr = 0;
oulijun9a443532016-07-21 19:06:38 +0800524 } else {
525 mr->type = MR_TYPE_MR;
Wei Hu(Xavier)ff795f72017-08-30 17:23:10 +0800526 if (!hr_dev->caps.pbl_hop_num) {
527 mr->pbl_buf = dma_alloc_coherent(dev, npages * 8,
528 &(mr->pbl_dma_addr),
529 GFP_KERNEL);
530 if (!mr->pbl_buf)
531 return -ENOMEM;
532 } else {
533 ret = hns_roce_mhop_alloc(hr_dev, npages, mr);
534 }
oulijun9a443532016-07-21 19:06:38 +0800535 }
536
Wei Hu(Xavier)ff795f72017-08-30 17:23:10 +0800537 return ret;
538}
539
540static void hns_roce_mhop_free(struct hns_roce_dev *hr_dev,
541 struct hns_roce_mr *mr)
542{
543 struct device *dev = hr_dev->dev;
544 int npages_allocated;
545 int npages;
546 int i, j;
547 u32 pbl_bt_sz;
548 u32 mhop_num;
549 u64 bt_idx;
550
551 npages = ib_umem_page_count(mr->umem);
552 pbl_bt_sz = 1 << (hr_dev->caps.pbl_ba_pg_sz + PAGE_SHIFT);
553 mhop_num = hr_dev->caps.pbl_hop_num;
554
555 if (mhop_num == HNS_ROCE_HOP_NUM_0)
556 return;
557
558 /* hop_num = 1 */
559 if (mhop_num == 1) {
560 dma_free_coherent(dev, (unsigned int)(npages * 8),
561 mr->pbl_buf, mr->pbl_dma_addr);
562 return;
563 }
564
565 dma_free_coherent(dev, pbl_bt_sz, mr->pbl_bt_l0,
566 mr->pbl_l0_dma_addr);
567
568 if (mhop_num == 2) {
569 for (i = 0; i < mr->l0_chunk_last_num; i++) {
570 if (i == mr->l0_chunk_last_num - 1) {
571 npages_allocated = i * (pbl_bt_sz / 8);
572
573 dma_free_coherent(dev,
574 (npages - npages_allocated) * 8,
575 mr->pbl_bt_l1[i],
576 mr->pbl_l1_dma_addr[i]);
577
578 break;
579 }
580
581 dma_free_coherent(dev, pbl_bt_sz, mr->pbl_bt_l1[i],
582 mr->pbl_l1_dma_addr[i]);
583 }
584 } else if (mhop_num == 3) {
585 for (i = 0; i < mr->l0_chunk_last_num; i++) {
586 dma_free_coherent(dev, pbl_bt_sz, mr->pbl_bt_l1[i],
587 mr->pbl_l1_dma_addr[i]);
588
589 for (j = 0; j < pbl_bt_sz / 8; j++) {
590 bt_idx = i * (pbl_bt_sz / 8) + j;
591
592 if ((i == mr->l0_chunk_last_num - 1)
593 && j == mr->l1_chunk_last_num - 1) {
594 npages_allocated = bt_idx *
595 (pbl_bt_sz / 8);
596
597 dma_free_coherent(dev,
598 (npages - npages_allocated) * 8,
599 mr->pbl_bt_l2[bt_idx],
600 mr->pbl_l2_dma_addr[bt_idx]);
601
602 break;
603 }
604
605 dma_free_coherent(dev, pbl_bt_sz,
606 mr->pbl_bt_l2[bt_idx],
607 mr->pbl_l2_dma_addr[bt_idx]);
608 }
609 }
610 }
611
612 kfree(mr->pbl_bt_l1);
613 kfree(mr->pbl_l1_dma_addr);
614 mr->pbl_bt_l1 = NULL;
615 mr->pbl_l1_dma_addr = NULL;
616 if (mhop_num == 3) {
617 kfree(mr->pbl_bt_l2);
618 kfree(mr->pbl_l2_dma_addr);
619 mr->pbl_bt_l2 = NULL;
620 mr->pbl_l2_dma_addr = NULL;
621 }
oulijun9a443532016-07-21 19:06:38 +0800622}
623
624static void hns_roce_mr_free(struct hns_roce_dev *hr_dev,
625 struct hns_roce_mr *mr)
626{
Wei Hu(Xavier)13ca9702017-08-30 17:23:02 +0800627 struct device *dev = hr_dev->dev;
oulijun9a443532016-07-21 19:06:38 +0800628 int npages = 0;
629 int ret;
630
631 if (mr->enabled) {
632 ret = hns_roce_hw2sw_mpt(hr_dev, NULL, key_to_hw_index(mr->key)
633 & (hr_dev->caps.num_mtpts - 1));
634 if (ret)
635 dev_warn(dev, "HW2SW_MPT failed (%d)\n", ret);
636 }
637
638 if (mr->size != ~0ULL) {
639 npages = ib_umem_page_count(mr->umem);
Wei Hu(Xavier)ff795f72017-08-30 17:23:10 +0800640
641 if (!hr_dev->caps.pbl_hop_num)
642 dma_free_coherent(dev, (unsigned int)(npages * 8),
643 mr->pbl_buf, mr->pbl_dma_addr);
644 else
645 hns_roce_mhop_free(hr_dev, mr);
oulijun9a443532016-07-21 19:06:38 +0800646 }
647
Wei Hu(Xavier)ff795f72017-08-30 17:23:10 +0800648 if (mr->enabled)
649 hns_roce_table_put(hr_dev, &hr_dev->mr_table.mtpt_table,
650 key_to_hw_index(mr->key));
651
oulijun9a443532016-07-21 19:06:38 +0800652 hns_roce_bitmap_free(&hr_dev->mr_table.mtpt_bitmap,
Wei Hu (Xavier)5e6ff782016-11-23 19:41:07 +0000653 key_to_hw_index(mr->key), BITMAP_NO_RR);
oulijun9a443532016-07-21 19:06:38 +0800654}
655
656static int hns_roce_mr_enable(struct hns_roce_dev *hr_dev,
657 struct hns_roce_mr *mr)
658{
659 int ret;
660 unsigned long mtpt_idx = key_to_hw_index(mr->key);
Wei Hu(Xavier)13ca9702017-08-30 17:23:02 +0800661 struct device *dev = hr_dev->dev;
oulijun9a443532016-07-21 19:06:38 +0800662 struct hns_roce_cmd_mailbox *mailbox;
663 struct hns_roce_mr_table *mr_table = &hr_dev->mr_table;
664
665 /* Prepare HEM entry memory */
666 ret = hns_roce_table_get(hr_dev, &mr_table->mtpt_table, mtpt_idx);
667 if (ret)
668 return ret;
669
670 /* Allocate mailbox memory */
671 mailbox = hns_roce_alloc_cmd_mailbox(hr_dev);
672 if (IS_ERR(mailbox)) {
673 ret = PTR_ERR(mailbox);
674 goto err_table;
675 }
676
677 ret = hr_dev->hw->write_mtpt(mailbox->buf, mr, mtpt_idx);
678 if (ret) {
679 dev_err(dev, "Write mtpt fail!\n");
680 goto err_page;
681 }
682
683 ret = hns_roce_sw2hw_mpt(hr_dev, mailbox,
684 mtpt_idx & (hr_dev->caps.num_mtpts - 1));
685 if (ret) {
686 dev_err(dev, "SW2HW_MPT failed (%d)\n", ret);
687 goto err_page;
688 }
689
690 mr->enabled = 1;
691 hns_roce_free_cmd_mailbox(hr_dev, mailbox);
692
693 return 0;
694
695err_page:
696 hns_roce_free_cmd_mailbox(hr_dev, mailbox);
697
698err_table:
699 hns_roce_table_put(hr_dev, &mr_table->mtpt_table, mtpt_idx);
700 return ret;
701}
702
703static int hns_roce_write_mtt_chunk(struct hns_roce_dev *hr_dev,
704 struct hns_roce_mtt *mtt, u32 start_index,
705 u32 npages, u64 *page_list)
706{
Shaobo Xu6a93c772017-08-30 17:23:08 +0800707 struct hns_roce_hem_table *table;
oulijun9a443532016-07-21 19:06:38 +0800708 dma_addr_t dma_handle;
Shaobo Xu6a93c772017-08-30 17:23:08 +0800709 __le64 *mtts;
oulijun9a443532016-07-21 19:06:38 +0800710 u32 s = start_index * sizeof(u64);
Wei Hu(Xavier)9a8982d2017-10-18 17:32:44 +0800711 u32 bt_page_size;
Shaobo Xu6a93c772017-08-30 17:23:08 +0800712 u32 i;
oulijun9a443532016-07-21 19:06:38 +0800713
Wei Hu(Xavier)9a8982d2017-10-18 17:32:44 +0800714 if (mtt->mtt_type == MTT_TYPE_WQE)
715 bt_page_size = 1 << (hr_dev->caps.mtt_ba_pg_sz + PAGE_SHIFT);
716 else
717 bt_page_size = 1 << (hr_dev->caps.cqe_ba_pg_sz + PAGE_SHIFT);
718
oulijun9a443532016-07-21 19:06:38 +0800719 /* All MTTs must fit in the same page */
Wei Hu(Xavier)9a8982d2017-10-18 17:32:44 +0800720 if (start_index / (bt_page_size / sizeof(u64)) !=
721 (start_index + npages - 1) / (bt_page_size / sizeof(u64)))
oulijun9a443532016-07-21 19:06:38 +0800722 return -EINVAL;
723
724 if (start_index & (HNS_ROCE_MTT_ENTRY_PER_SEG - 1))
725 return -EINVAL;
726
Shaobo Xu9766edc2017-08-30 17:23:09 +0800727 if (mtt->mtt_type == MTT_TYPE_WQE)
728 table = &hr_dev->mr_table.mtt_table;
729 else
730 table = &hr_dev->mr_table.mtt_cqe_table;
731
Shaobo Xu6a93c772017-08-30 17:23:08 +0800732 mtts = hns_roce_table_find(hr_dev, table,
oulijun9a443532016-07-21 19:06:38 +0800733 mtt->first_seg + s / hr_dev->caps.mtt_entry_sz,
734 &dma_handle);
735 if (!mtts)
736 return -ENOMEM;
737
738 /* Save page addr, low 12 bits : 0 */
Shaobo Xu6a93c772017-08-30 17:23:08 +0800739 for (i = 0; i < npages; ++i) {
740 if (!hr_dev->caps.mtt_hop_num)
741 mtts[i] = cpu_to_le64(page_list[i] >> PAGE_ADDR_SHIFT);
742 else
743 mtts[i] = cpu_to_le64(page_list[i]);
744 }
oulijun9a443532016-07-21 19:06:38 +0800745
746 return 0;
747}
748
749static int hns_roce_write_mtt(struct hns_roce_dev *hr_dev,
750 struct hns_roce_mtt *mtt, u32 start_index,
751 u32 npages, u64 *page_list)
752{
753 int chunk;
754 int ret;
Wei Hu(Xavier)9a8982d2017-10-18 17:32:44 +0800755 u32 bt_page_size;
oulijun9a443532016-07-21 19:06:38 +0800756
757 if (mtt->order < 0)
758 return -EINVAL;
759
Wei Hu(Xavier)9a8982d2017-10-18 17:32:44 +0800760 if (mtt->mtt_type == MTT_TYPE_WQE)
761 bt_page_size = 1 << (hr_dev->caps.mtt_ba_pg_sz + PAGE_SHIFT);
762 else
763 bt_page_size = 1 << (hr_dev->caps.cqe_ba_pg_sz + PAGE_SHIFT);
764
oulijun9a443532016-07-21 19:06:38 +0800765 while (npages > 0) {
Wei Hu(Xavier)9a8982d2017-10-18 17:32:44 +0800766 chunk = min_t(int, bt_page_size / sizeof(u64), npages);
oulijun9a443532016-07-21 19:06:38 +0800767
768 ret = hns_roce_write_mtt_chunk(hr_dev, mtt, start_index, chunk,
769 page_list);
770 if (ret)
771 return ret;
772
773 npages -= chunk;
774 start_index += chunk;
775 page_list += chunk;
776 }
777
778 return 0;
779}
780
781int hns_roce_buf_write_mtt(struct hns_roce_dev *hr_dev,
782 struct hns_roce_mtt *mtt, struct hns_roce_buf *buf)
783{
Shaobo Xu9766edc2017-08-30 17:23:09 +0800784 u64 *page_list;
785 int ret;
786 u32 i;
oulijun9a443532016-07-21 19:06:38 +0800787
788 page_list = kmalloc_array(buf->npages, sizeof(*page_list), GFP_KERNEL);
789 if (!page_list)
790 return -ENOMEM;
791
792 for (i = 0; i < buf->npages; ++i) {
793 if (buf->nbufs == 1)
794 page_list[i] = buf->direct.map + (i << buf->page_shift);
795 else
796 page_list[i] = buf->page_list[i].map;
797
798 }
799 ret = hns_roce_write_mtt(hr_dev, mtt, 0, buf->npages, page_list);
800
801 kfree(page_list);
802
803 return ret;
804}
805
806int hns_roce_init_mr_table(struct hns_roce_dev *hr_dev)
807{
808 struct hns_roce_mr_table *mr_table = &hr_dev->mr_table;
Shaobo Xu9766edc2017-08-30 17:23:09 +0800809 int ret;
oulijun9a443532016-07-21 19:06:38 +0800810
811 ret = hns_roce_bitmap_init(&mr_table->mtpt_bitmap,
812 hr_dev->caps.num_mtpts,
813 hr_dev->caps.num_mtpts - 1,
814 hr_dev->caps.reserved_mrws, 0);
815 if (ret)
816 return ret;
817
818 ret = hns_roce_buddy_init(&mr_table->mtt_buddy,
819 ilog2(hr_dev->caps.num_mtt_segs));
820 if (ret)
821 goto err_buddy;
822
Shaobo Xu9766edc2017-08-30 17:23:09 +0800823 if (hns_roce_check_whether_mhop(hr_dev, HEM_TYPE_CQE)) {
824 ret = hns_roce_buddy_init(&mr_table->mtt_cqe_buddy,
825 ilog2(hr_dev->caps.num_cqe_segs));
826 if (ret)
827 goto err_buddy_cqe;
828 }
oulijun9a443532016-07-21 19:06:38 +0800829 return 0;
830
Shaobo Xu9766edc2017-08-30 17:23:09 +0800831err_buddy_cqe:
832 hns_roce_buddy_cleanup(&mr_table->mtt_buddy);
833
oulijun9a443532016-07-21 19:06:38 +0800834err_buddy:
835 hns_roce_bitmap_cleanup(&mr_table->mtpt_bitmap);
836 return ret;
837}
838
839void hns_roce_cleanup_mr_table(struct hns_roce_dev *hr_dev)
840{
841 struct hns_roce_mr_table *mr_table = &hr_dev->mr_table;
842
843 hns_roce_buddy_cleanup(&mr_table->mtt_buddy);
Shaobo Xu9766edc2017-08-30 17:23:09 +0800844 if (hns_roce_check_whether_mhop(hr_dev, HEM_TYPE_CQE))
845 hns_roce_buddy_cleanup(&mr_table->mtt_cqe_buddy);
oulijun9a443532016-07-21 19:06:38 +0800846 hns_roce_bitmap_cleanup(&mr_table->mtpt_bitmap);
847}
848
849struct ib_mr *hns_roce_get_dma_mr(struct ib_pd *pd, int acc)
850{
Wei Hu(Xavier)ff795f72017-08-30 17:23:10 +0800851 struct hns_roce_mr *mr;
852 int ret;
oulijun9a443532016-07-21 19:06:38 +0800853
854 mr = kmalloc(sizeof(*mr), GFP_KERNEL);
855 if (mr == NULL)
856 return ERR_PTR(-ENOMEM);
857
858 /* Allocate memory region key */
859 ret = hns_roce_mr_alloc(to_hr_dev(pd->device), to_hr_pd(pd)->pdn, 0,
860 ~0ULL, acc, 0, mr);
861 if (ret)
862 goto err_free;
863
864 ret = hns_roce_mr_enable(to_hr_dev(pd->device), mr);
865 if (ret)
866 goto err_mr;
867
868 mr->ibmr.rkey = mr->ibmr.lkey = mr->key;
869 mr->umem = NULL;
870
871 return &mr->ibmr;
872
873err_mr:
874 hns_roce_mr_free(to_hr_dev(pd->device), mr);
875
876err_free:
877 kfree(mr);
878 return ERR_PTR(ret);
879}
880
881int hns_roce_ib_umem_write_mtt(struct hns_roce_dev *hr_dev,
882 struct hns_roce_mtt *mtt, struct ib_umem *umem)
883{
Wei Hu(Xavier)9a8982d2017-10-18 17:32:44 +0800884 struct device *dev = hr_dev->dev;
oulijun9a443532016-07-21 19:06:38 +0800885 struct scatterlist *sg;
Wei Hu(Xavier)9a8982d2017-10-18 17:32:44 +0800886 unsigned int order;
oulijun9a443532016-07-21 19:06:38 +0800887 int i, k, entry;
Wei Hu(Xavier)9a8982d2017-10-18 17:32:44 +0800888 int npage = 0;
oulijun9a443532016-07-21 19:06:38 +0800889 int ret = 0;
oulijun9a443532016-07-21 19:06:38 +0800890 int len;
Wei Hu(Xavier)9a8982d2017-10-18 17:32:44 +0800891 u64 page_addr;
892 u64 *pages;
893 u32 bt_page_size;
894 u32 n;
oulijun9a443532016-07-21 19:06:38 +0800895
Wei Hu(Xavier)9a8982d2017-10-18 17:32:44 +0800896 order = mtt->mtt_type == MTT_TYPE_WQE ? hr_dev->caps.mtt_ba_pg_sz :
897 hr_dev->caps.cqe_ba_pg_sz;
898 bt_page_size = 1 << (order + PAGE_SHIFT);
899
900 pages = (u64 *) __get_free_pages(GFP_KERNEL, order);
oulijun9a443532016-07-21 19:06:38 +0800901 if (!pages)
902 return -ENOMEM;
903
904 i = n = 0;
905
906 for_each_sg(umem->sg_head.sgl, sg, umem->nmap, entry) {
Wei Hu(Xavier)9a8982d2017-10-18 17:32:44 +0800907 len = sg_dma_len(sg) >> PAGE_SHIFT;
oulijun9a443532016-07-21 19:06:38 +0800908 for (k = 0; k < len; ++k) {
Wei Hu(Xavier)9a8982d2017-10-18 17:32:44 +0800909 page_addr =
910 sg_dma_address(sg) + (k << umem->page_shift);
911 if (!(npage % (1 << (mtt->page_shift - PAGE_SHIFT)))) {
912 if (page_addr & ((1 << mtt->page_shift) - 1)) {
913 dev_err(dev, "page_addr 0x%llx is not page_shift %d alignment!\n",
914 page_addr, mtt->page_shift);
915 ret = -EINVAL;
916 goto out;
917 }
918 pages[i++] = page_addr;
919 }
920 npage++;
921 if (i == bt_page_size / sizeof(u64)) {
oulijun9a443532016-07-21 19:06:38 +0800922 ret = hns_roce_write_mtt(hr_dev, mtt, n, i,
923 pages);
924 if (ret)
925 goto out;
926 n += i;
927 i = 0;
928 }
929 }
930 }
931
932 if (i)
933 ret = hns_roce_write_mtt(hr_dev, mtt, n, i, pages);
934
935out:
oulijund480bb52018-02-07 17:49:30 +0800936 free_pages((unsigned long) pages, order);
oulijun9a443532016-07-21 19:06:38 +0800937 return ret;
938}
939
Wei Hu(Xavier)ff795f72017-08-30 17:23:10 +0800940static int hns_roce_ib_umem_write_mr(struct hns_roce_dev *hr_dev,
941 struct hns_roce_mr *mr,
oulijun9a443532016-07-21 19:06:38 +0800942 struct ib_umem *umem)
943{
oulijun9a443532016-07-21 19:06:38 +0800944 struct scatterlist *sg;
Wei Hu(Xavier)9a8982d2017-10-18 17:32:44 +0800945 int i = 0, j = 0, k;
Wei Hu(Xavier)ff795f72017-08-30 17:23:10 +0800946 int entry;
Wei Hu(Xavier)9a8982d2017-10-18 17:32:44 +0800947 int len;
948 u64 page_addr;
949 u32 pbl_bt_sz;
Wei Hu(Xavier)ff795f72017-08-30 17:23:10 +0800950
951 if (hr_dev->caps.pbl_hop_num == HNS_ROCE_HOP_NUM_0)
952 return 0;
oulijun9a443532016-07-21 19:06:38 +0800953
Wei Hu(Xavier)9a8982d2017-10-18 17:32:44 +0800954 pbl_bt_sz = 1 << (hr_dev->caps.pbl_ba_pg_sz + PAGE_SHIFT);
oulijun9a443532016-07-21 19:06:38 +0800955 for_each_sg(umem->sg_head.sgl, sg, umem->nmap, entry) {
Wei Hu(Xavier)9a8982d2017-10-18 17:32:44 +0800956 len = sg_dma_len(sg) >> PAGE_SHIFT;
957 for (k = 0; k < len; ++k) {
958 page_addr = sg_dma_address(sg) +
959 (k << umem->page_shift);
Wei Hu(Xavier)ff795f72017-08-30 17:23:10 +0800960
Wei Hu(Xavier)9a8982d2017-10-18 17:32:44 +0800961 if (!hr_dev->caps.pbl_hop_num) {
962 mr->pbl_buf[i++] = page_addr >> 12;
963 } else if (hr_dev->caps.pbl_hop_num == 1) {
964 mr->pbl_buf[i++] = page_addr;
965 } else {
966 if (hr_dev->caps.pbl_hop_num == 2)
967 mr->pbl_bt_l1[i][j] = page_addr;
968 else if (hr_dev->caps.pbl_hop_num == 3)
969 mr->pbl_bt_l2[i][j] = page_addr;
970
971 j++;
972 if (j >= (pbl_bt_sz / 8)) {
973 i++;
974 j = 0;
975 }
Wei Hu(Xavier)ff795f72017-08-30 17:23:10 +0800976 }
977 }
oulijun9a443532016-07-21 19:06:38 +0800978 }
979
980 /* Memory barrier */
981 mb();
982
983 return 0;
984}
985
986struct ib_mr *hns_roce_reg_user_mr(struct ib_pd *pd, u64 start, u64 length,
987 u64 virt_addr, int access_flags,
988 struct ib_udata *udata)
989{
990 struct hns_roce_dev *hr_dev = to_hr_dev(pd->device);
Wei Hu(Xavier)13ca9702017-08-30 17:23:02 +0800991 struct device *dev = hr_dev->dev;
Wei Hu(Xavier)ff795f72017-08-30 17:23:10 +0800992 struct hns_roce_mr *mr;
993 int bt_size;
994 int ret;
995 int n;
996 int i;
oulijun9a443532016-07-21 19:06:38 +0800997
998 mr = kmalloc(sizeof(*mr), GFP_KERNEL);
999 if (!mr)
1000 return ERR_PTR(-ENOMEM);
1001
1002 mr->umem = ib_umem_get(pd->uobject->context, start, length,
1003 access_flags, 0);
1004 if (IS_ERR(mr->umem)) {
1005 ret = PTR_ERR(mr->umem);
1006 goto err_free;
1007 }
1008
1009 n = ib_umem_page_count(mr->umem);
Artemy Kovalyov3e7e1192017-04-05 09:23:50 +03001010 if (mr->umem->page_shift != HNS_ROCE_HEM_PAGE_SHIFT) {
1011 dev_err(dev, "Just support 4K page size but is 0x%lx now!\n",
1012 BIT(mr->umem->page_shift));
Lijun Ou1cd11062016-09-20 17:07:03 +01001013 ret = -EINVAL;
1014 goto err_umem;
oulijun9a443532016-07-21 19:06:38 +08001015 }
1016
Wei Hu(Xavier)ff795f72017-08-30 17:23:10 +08001017 if (!hr_dev->caps.pbl_hop_num) {
1018 if (n > HNS_ROCE_MAX_MTPT_PBL_NUM) {
1019 dev_err(dev,
1020 " MR len %lld err. MR is limited to 4G at most!\n",
1021 length);
1022 ret = -EINVAL;
1023 goto err_umem;
1024 }
1025 } else {
1026 int pbl_size = 1;
1027
Wei Hu(Xavier)9a8982d2017-10-18 17:32:44 +08001028 bt_size = (1 << (hr_dev->caps.pbl_ba_pg_sz + PAGE_SHIFT)) / 8;
Wei Hu(Xavier)ff795f72017-08-30 17:23:10 +08001029 for (i = 0; i < hr_dev->caps.pbl_hop_num; i++)
1030 pbl_size *= bt_size;
1031 if (n > pbl_size) {
1032 dev_err(dev,
1033 " MR len %lld err. MR page num is limited to %d!\n",
1034 length, pbl_size);
1035 ret = -EINVAL;
1036 goto err_umem;
1037 }
oulijun9a443532016-07-21 19:06:38 +08001038 }
1039
1040 ret = hns_roce_mr_alloc(hr_dev, to_hr_pd(pd)->pdn, virt_addr, length,
1041 access_flags, n, mr);
1042 if (ret)
1043 goto err_umem;
1044
Wei Hu(Xavier)ff795f72017-08-30 17:23:10 +08001045 ret = hns_roce_ib_umem_write_mr(hr_dev, mr, mr->umem);
oulijun9a443532016-07-21 19:06:38 +08001046 if (ret)
1047 goto err_mr;
1048
1049 ret = hns_roce_mr_enable(hr_dev, mr);
1050 if (ret)
1051 goto err_mr;
1052
1053 mr->ibmr.rkey = mr->ibmr.lkey = mr->key;
1054
1055 return &mr->ibmr;
1056
1057err_mr:
1058 hns_roce_mr_free(hr_dev, mr);
1059
1060err_umem:
1061 ib_umem_release(mr->umem);
1062
1063err_free:
1064 kfree(mr);
1065 return ERR_PTR(ret);
1066}
1067
Wei Hu(Xavier)a2c80b72017-10-26 17:10:23 +08001068int hns_roce_rereg_user_mr(struct ib_mr *ibmr, int flags, u64 start, u64 length,
1069 u64 virt_addr, int mr_access_flags, struct ib_pd *pd,
1070 struct ib_udata *udata)
1071{
1072 struct hns_roce_dev *hr_dev = to_hr_dev(ibmr->device);
1073 struct hns_roce_mr *mr = to_hr_mr(ibmr);
1074 struct hns_roce_cmd_mailbox *mailbox;
1075 struct device *dev = hr_dev->dev;
1076 unsigned long mtpt_idx;
1077 u32 pdn = 0;
1078 int npages;
1079 int ret;
1080
1081 if (!mr->enabled)
1082 return -EINVAL;
1083
1084 mailbox = hns_roce_alloc_cmd_mailbox(hr_dev);
1085 if (IS_ERR(mailbox))
1086 return PTR_ERR(mailbox);
1087
1088 mtpt_idx = key_to_hw_index(mr->key) & (hr_dev->caps.num_mtpts - 1);
1089 ret = hns_roce_cmd_mbox(hr_dev, 0, mailbox->dma, mtpt_idx, 0,
1090 HNS_ROCE_CMD_QUERY_MPT,
1091 HNS_ROCE_CMD_TIMEOUT_MSECS);
1092 if (ret)
1093 goto free_cmd_mbox;
1094
1095 ret = hns_roce_hw2sw_mpt(hr_dev, NULL, mtpt_idx);
1096 if (ret)
1097 dev_warn(dev, "HW2SW_MPT failed (%d)\n", ret);
1098
1099 mr->enabled = 0;
1100
1101 if (flags & IB_MR_REREG_PD)
1102 pdn = to_hr_pd(pd)->pdn;
1103
1104 if (flags & IB_MR_REREG_TRANS) {
1105 if (mr->size != ~0ULL) {
1106 npages = ib_umem_page_count(mr->umem);
1107
1108 if (hr_dev->caps.pbl_hop_num)
1109 hns_roce_mhop_free(hr_dev, mr);
1110 else
1111 dma_free_coherent(dev, npages * 8, mr->pbl_buf,
1112 mr->pbl_dma_addr);
1113 }
1114 ib_umem_release(mr->umem);
1115
1116 mr->umem = ib_umem_get(ibmr->uobject->context, start, length,
1117 mr_access_flags, 0);
1118 if (IS_ERR(mr->umem)) {
1119 ret = PTR_ERR(mr->umem);
1120 mr->umem = NULL;
1121 goto free_cmd_mbox;
1122 }
1123 npages = ib_umem_page_count(mr->umem);
1124
1125 if (hr_dev->caps.pbl_hop_num) {
1126 ret = hns_roce_mhop_alloc(hr_dev, npages, mr);
1127 if (ret)
1128 goto release_umem;
1129 } else {
1130 mr->pbl_buf = dma_alloc_coherent(dev, npages * 8,
1131 &(mr->pbl_dma_addr),
1132 GFP_KERNEL);
1133 if (!mr->pbl_buf) {
1134 ret = -ENOMEM;
1135 goto release_umem;
1136 }
1137 }
1138 }
1139
1140 ret = hr_dev->hw->rereg_write_mtpt(hr_dev, mr, flags, pdn,
1141 mr_access_flags, virt_addr,
1142 length, mailbox->buf);
1143 if (ret) {
1144 if (flags & IB_MR_REREG_TRANS)
1145 goto release_umem;
1146 else
1147 goto free_cmd_mbox;
1148 }
1149
1150 if (flags & IB_MR_REREG_TRANS) {
1151 ret = hns_roce_ib_umem_write_mr(hr_dev, mr, mr->umem);
1152 if (ret) {
1153 if (mr->size != ~0ULL) {
1154 npages = ib_umem_page_count(mr->umem);
1155
1156 if (hr_dev->caps.pbl_hop_num)
1157 hns_roce_mhop_free(hr_dev, mr);
1158 else
1159 dma_free_coherent(dev, npages * 8,
1160 mr->pbl_buf,
1161 mr->pbl_dma_addr);
1162 }
1163
1164 goto release_umem;
1165 }
1166 }
1167
1168 ret = hns_roce_sw2hw_mpt(hr_dev, mailbox, mtpt_idx);
1169 if (ret) {
1170 dev_err(dev, "SW2HW_MPT failed (%d)\n", ret);
1171 goto release_umem;
1172 }
1173
1174 mr->enabled = 1;
1175 if (flags & IB_MR_REREG_ACCESS)
1176 mr->access = mr_access_flags;
1177
1178 hns_roce_free_cmd_mailbox(hr_dev, mailbox);
1179
1180 return 0;
1181
1182release_umem:
1183 ib_umem_release(mr->umem);
1184
1185free_cmd_mbox:
1186 hns_roce_free_cmd_mailbox(hr_dev, mailbox);
1187
1188 return ret;
1189}
1190
oulijun9a443532016-07-21 19:06:38 +08001191int hns_roce_dereg_mr(struct ib_mr *ibmr)
1192{
Shaobo Xubfcc6812016-11-29 23:10:26 +00001193 struct hns_roce_dev *hr_dev = to_hr_dev(ibmr->device);
oulijun9a443532016-07-21 19:06:38 +08001194 struct hns_roce_mr *mr = to_hr_mr(ibmr);
Shaobo Xubfcc6812016-11-29 23:10:26 +00001195 int ret = 0;
oulijun9a443532016-07-21 19:06:38 +08001196
Shaobo Xubfcc6812016-11-29 23:10:26 +00001197 if (hr_dev->hw->dereg_mr) {
1198 ret = hr_dev->hw->dereg_mr(hr_dev, mr);
1199 } else {
1200 hns_roce_mr_free(hr_dev, mr);
oulijun9a443532016-07-21 19:06:38 +08001201
Shaobo Xubfcc6812016-11-29 23:10:26 +00001202 if (mr->umem)
1203 ib_umem_release(mr->umem);
oulijun9a443532016-07-21 19:06:38 +08001204
Shaobo Xubfcc6812016-11-29 23:10:26 +00001205 kfree(mr);
1206 }
1207
1208 return ret;
oulijun9a443532016-07-21 19:06:38 +08001209}