1 /*
2 * Copyright (c) 2013-2015, Mellanox Technologies. All rights reserved.
3 *
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:
9 *
10 * Redistribution and use in source and binary forms, with or
11 * without modification, are permitted provided that the following
12 * conditions are met:
13 *
14 * - Redistributions of source code must retain the above
15 * copyright notice, this list of conditions and the following
16 * disclaimer.
17 *
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.
22 *
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
30 * SOFTWARE.
31 */
32
33 #include <linux/highmem.h>
34 #include <linux/kernel.h>
35 #include <linux/delay.h>
36 #include <linux/mlx5/driver.h>
37 #include <linux/xarray.h>
38 #include "mlx5_core.h"
39 #include "lib/eq.h"
40 #include "lib/tout.h"
41
42 enum {
43 MLX5_PAGES_CANT_GIVE = 0,
44 MLX5_PAGES_GIVE = 1,
45 MLX5_PAGES_TAKE = 2
46 };
47
48 struct mlx5_pages_req {
49 struct mlx5_core_dev *dev;
50 u16 func_id;
51 u8 ec_function;
52 s32 npages;
53 struct work_struct work;
54 u8 release_all;
55 };
56
57 struct fw_page {
58 struct rb_node rb_node;
59 u64 addr;
60 struct page *page;
61 u32 function;
62 unsigned long bitmask;
63 struct list_head list;
64 unsigned int free_count;
65 };
66
67 enum {
68 MLX5_MAX_RECLAIM_TIME_MILI = 5000,
69 MLX5_NUM_4K_IN_PAGE = PAGE_SIZE / MLX5_ADAPTER_PAGE_SIZE,
70 };
71
get_function(u16 func_id,bool ec_function)72 static u32 get_function(u16 func_id, bool ec_function)
73 {
74 return (u32)func_id | (ec_function << 16);
75 }
76
func_id_to_type(struct mlx5_core_dev * dev,u16 func_id,bool ec_function)77 static u16 func_id_to_type(struct mlx5_core_dev *dev, u16 func_id, bool ec_function)
78 {
79 if (!func_id)
80 return mlx5_core_is_ecpf(dev) && !ec_function ? MLX5_HOST_PF : MLX5_PF;
81
82 if (func_id <= max(mlx5_core_max_vfs(dev), mlx5_core_max_ec_vfs(dev))) {
83 if (ec_function)
84 return MLX5_EC_VF;
85 else
86 return MLX5_VF;
87 }
88 return MLX5_SF;
89 }
90
mlx5_get_ec_function(u32 function)91 static u32 mlx5_get_ec_function(u32 function)
92 {
93 return function >> 16;
94 }
95
mlx5_get_func_id(u32 function)96 static u32 mlx5_get_func_id(u32 function)
97 {
98 return function & 0xffff;
99 }
100
page_root_per_function(struct mlx5_core_dev * dev,u32 function)101 static struct rb_root *page_root_per_function(struct mlx5_core_dev *dev, u32 function)
102 {
103 struct rb_root *root;
104 int err;
105
106 root = xa_load(&dev->priv.page_root_xa, function);
107 if (root)
108 return root;
109
110 root = kzalloc(sizeof(*root), GFP_KERNEL);
111 if (!root)
112 return ERR_PTR(-ENOMEM);
113
114 err = xa_insert(&dev->priv.page_root_xa, function, root, GFP_KERNEL);
115 if (err) {
116 kfree(root);
117 return ERR_PTR(err);
118 }
119
120 *root = RB_ROOT;
121
122 return root;
123 }
124
insert_page(struct mlx5_core_dev * dev,u64 addr,struct page * page,u32 function)125 static int insert_page(struct mlx5_core_dev *dev, u64 addr, struct page *page, u32 function)
126 {
127 struct rb_node *parent = NULL;
128 struct rb_root *root;
129 struct rb_node **new;
130 struct fw_page *nfp;
131 struct fw_page *tfp;
132 int i;
133
134 root = page_root_per_function(dev, function);
135 if (IS_ERR(root))
136 return PTR_ERR(root);
137
138 new = &root->rb_node;
139
140 while (*new) {
141 parent = *new;
142 tfp = rb_entry(parent, struct fw_page, rb_node);
143 if (tfp->addr < addr)
144 new = &parent->rb_left;
145 else if (tfp->addr > addr)
146 new = &parent->rb_right;
147 else
148 return -EEXIST;
149 }
150
151 nfp = kzalloc(sizeof(*nfp), GFP_KERNEL);
152 if (!nfp)
153 return -ENOMEM;
154
155 nfp->addr = addr;
156 nfp->page = page;
157 nfp->function = function;
158 nfp->free_count = MLX5_NUM_4K_IN_PAGE;
159 for (i = 0; i < MLX5_NUM_4K_IN_PAGE; i++)
160 set_bit(i, &nfp->bitmask);
161
162 rb_link_node(&nfp->rb_node, parent, new);
163 rb_insert_color(&nfp->rb_node, root);
164 list_add(&nfp->list, &dev->priv.free_list);
165
166 return 0;
167 }
168
find_fw_page(struct mlx5_core_dev * dev,u64 addr,u32 function)169 static struct fw_page *find_fw_page(struct mlx5_core_dev *dev, u64 addr,
170 u32 function)
171 {
172 struct fw_page *result = NULL;
173 struct rb_root *root;
174 struct rb_node *tmp;
175 struct fw_page *tfp;
176
177 root = xa_load(&dev->priv.page_root_xa, function);
178 if (WARN_ON_ONCE(!root))
179 return NULL;
180
181 tmp = root->rb_node;
182
183 while (tmp) {
184 tfp = rb_entry(tmp, struct fw_page, rb_node);
185 if (tfp->addr < addr) {
186 tmp = tmp->rb_left;
187 } else if (tfp->addr > addr) {
188 tmp = tmp->rb_right;
189 } else {
190 result = tfp;
191 break;
192 }
193 }
194
195 return result;
196 }
197
mlx5_cmd_query_pages(struct mlx5_core_dev * dev,u16 * func_id,s32 * npages,int boot)198 static int mlx5_cmd_query_pages(struct mlx5_core_dev *dev, u16 *func_id,
199 s32 *npages, int boot)
200 {
201 u32 out[MLX5_ST_SZ_DW(query_pages_out)] = {};
202 u32 in[MLX5_ST_SZ_DW(query_pages_in)] = {};
203 int err;
204
205 MLX5_SET(query_pages_in, in, opcode, MLX5_CMD_OP_QUERY_PAGES);
206 MLX5_SET(query_pages_in, in, op_mod, boot ?
207 MLX5_QUERY_PAGES_IN_OP_MOD_BOOT_PAGES :
208 MLX5_QUERY_PAGES_IN_OP_MOD_INIT_PAGES);
209 MLX5_SET(query_pages_in, in, embedded_cpu_function, mlx5_core_is_ecpf(dev));
210
211 err = mlx5_cmd_exec_inout(dev, query_pages, in, out);
212 if (err)
213 return err;
214
215 *npages = MLX5_GET(query_pages_out, out, num_pages);
216 *func_id = MLX5_GET(query_pages_out, out, function_id);
217
218 return err;
219 }
220
alloc_4k(struct mlx5_core_dev * dev,u64 * addr,u32 function)221 static int alloc_4k(struct mlx5_core_dev *dev, u64 *addr, u32 function)
222 {
223 struct fw_page *fp = NULL;
224 struct fw_page *iter;
225 unsigned n;
226
227 list_for_each_entry(iter, &dev->priv.free_list, list) {
228 if (iter->function != function)
229 continue;
230 fp = iter;
231 }
232
233 if (list_empty(&dev->priv.free_list) || !fp)
234 return -ENOMEM;
235
236 n = find_first_bit(&fp->bitmask, 8 * sizeof(fp->bitmask));
237 if (n >= MLX5_NUM_4K_IN_PAGE) {
238 mlx5_core_warn(dev, "alloc 4k bug: fw page = 0x%llx, n = %u, bitmask: %lu, max num of 4K pages: %d\n",
239 fp->addr, n, fp->bitmask, MLX5_NUM_4K_IN_PAGE);
240 return -ENOENT;
241 }
242 clear_bit(n, &fp->bitmask);
243 fp->free_count--;
244 if (!fp->free_count)
245 list_del(&fp->list);
246
247 *addr = fp->addr + n * MLX5_ADAPTER_PAGE_SIZE;
248
249 return 0;
250 }
251
252 #define MLX5_U64_4K_PAGE_MASK ((~(u64)0U) << PAGE_SHIFT)
253
free_fwp(struct mlx5_core_dev * dev,struct fw_page * fwp,bool in_free_list)254 static void free_fwp(struct mlx5_core_dev *dev, struct fw_page *fwp,
255 bool in_free_list)
256 {
257 struct rb_root *root;
258
259 root = xa_load(&dev->priv.page_root_xa, fwp->function);
260 if (WARN_ON_ONCE(!root))
261 return;
262
263 rb_erase(&fwp->rb_node, root);
264 if (in_free_list)
265 list_del(&fwp->list);
266 dma_unmap_page(mlx5_core_dma_dev(dev), fwp->addr & MLX5_U64_4K_PAGE_MASK,
267 PAGE_SIZE, DMA_BIDIRECTIONAL);
268 __free_page(fwp->page);
269 kfree(fwp);
270 }
271
free_4k(struct mlx5_core_dev * dev,u64 addr,u32 function)272 static void free_4k(struct mlx5_core_dev *dev, u64 addr, u32 function)
273 {
274 struct fw_page *fwp;
275 int n;
276
277 fwp = find_fw_page(dev, addr & MLX5_U64_4K_PAGE_MASK, function);
278 if (!fwp) {
279 mlx5_core_warn_rl(dev, "page not found\n");
280 return;
281 }
282 n = (addr & ~MLX5_U64_4K_PAGE_MASK) >> MLX5_ADAPTER_PAGE_SHIFT;
283 fwp->free_count++;
284 set_bit(n, &fwp->bitmask);
285 if (fwp->free_count == MLX5_NUM_4K_IN_PAGE)
286 free_fwp(dev, fwp, fwp->free_count != 1);
287 else if (fwp->free_count == 1)
288 list_add(&fwp->list, &dev->priv.free_list);
289 }
290
alloc_system_page(struct mlx5_core_dev * dev,u32 function)291 static int alloc_system_page(struct mlx5_core_dev *dev, u32 function)
292 {
293 struct device *device = mlx5_core_dma_dev(dev);
294 int nid = dev_to_node(device);
295 struct page *page;
296 u64 zero_addr = 1;
297 u64 addr;
298 int err;
299
300 page = alloc_pages_node(nid, GFP_HIGHUSER, 0);
301 if (!page) {
302 mlx5_core_warn(dev, "failed to allocate page\n");
303 return -ENOMEM;
304 }
305 map:
306 addr = dma_map_page(device, page, 0, PAGE_SIZE, DMA_BIDIRECTIONAL);
307 if (dma_mapping_error(device, addr)) {
308 mlx5_core_warn(dev, "failed dma mapping page\n");
309 err = -ENOMEM;
310 goto err_mapping;
311 }
312
313 /* Firmware doesn't support page with physical address 0 */
314 if (addr == 0) {
315 zero_addr = addr;
316 goto map;
317 }
318
319 err = insert_page(dev, addr, page, function);
320 if (err) {
321 mlx5_core_err(dev, "failed to track allocated page\n");
322 dma_unmap_page(device, addr, PAGE_SIZE, DMA_BIDIRECTIONAL);
323 }
324
325 err_mapping:
326 if (err)
327 __free_page(page);
328
329 if (zero_addr == 0)
330 dma_unmap_page(device, zero_addr, PAGE_SIZE,
331 DMA_BIDIRECTIONAL);
332
333 return err;
334 }
335
page_notify_fail(struct mlx5_core_dev * dev,u16 func_id,bool ec_function)336 static void page_notify_fail(struct mlx5_core_dev *dev, u16 func_id,
337 bool ec_function)
338 {
339 u32 in[MLX5_ST_SZ_DW(manage_pages_in)] = {};
340 int err;
341
342 MLX5_SET(manage_pages_in, in, opcode, MLX5_CMD_OP_MANAGE_PAGES);
343 MLX5_SET(manage_pages_in, in, op_mod, MLX5_PAGES_CANT_GIVE);
344 MLX5_SET(manage_pages_in, in, function_id, func_id);
345 MLX5_SET(manage_pages_in, in, embedded_cpu_function, ec_function);
346
347 err = mlx5_cmd_exec_in(dev, manage_pages, in);
348 if (err)
349 mlx5_core_warn(dev, "page notify failed func_id(%d) err(%d)\n",
350 func_id, err);
351 }
352
give_pages(struct mlx5_core_dev * dev,u16 func_id,int npages,int event,bool ec_function)353 static int give_pages(struct mlx5_core_dev *dev, u16 func_id, int npages,
354 int event, bool ec_function)
355 {
356 u32 function = get_function(func_id, ec_function);
357 u32 out[MLX5_ST_SZ_DW(manage_pages_out)] = {0};
358 int inlen = MLX5_ST_SZ_BYTES(manage_pages_in);
359 int notify_fail = event;
360 u16 func_type;
361 u64 addr;
362 int err;
363 u32 *in;
364 int i;
365
366 inlen += npages * MLX5_FLD_SZ_BYTES(manage_pages_in, pas[0]);
367 in = kvzalloc(inlen, GFP_KERNEL);
368 if (!in) {
369 err = -ENOMEM;
370 mlx5_core_warn(dev, "vzalloc failed %d\n", inlen);
371 goto out_free;
372 }
373
374 for (i = 0; i < npages; i++) {
375 retry:
376 err = alloc_4k(dev, &addr, function);
377 if (err) {
378 if (err == -ENOMEM)
379 err = alloc_system_page(dev, function);
380 if (err) {
381 dev->priv.fw_pages_alloc_failed += (npages - i);
382 goto out_4k;
383 }
384
385 goto retry;
386 }
387 MLX5_ARRAY_SET64(manage_pages_in, in, pas, i, addr);
388 }
389
390 MLX5_SET(manage_pages_in, in, opcode, MLX5_CMD_OP_MANAGE_PAGES);
391 MLX5_SET(manage_pages_in, in, op_mod, MLX5_PAGES_GIVE);
392 MLX5_SET(manage_pages_in, in, function_id, func_id);
393 MLX5_SET(manage_pages_in, in, input_num_entries, npages);
394 MLX5_SET(manage_pages_in, in, embedded_cpu_function, ec_function);
395
396 err = mlx5_cmd_do(dev, in, inlen, out, sizeof(out));
397 if (err == -EREMOTEIO) {
398 notify_fail = 0;
399 /* if triggered by FW and failed by FW ignore */
400 if (event) {
401 err = 0;
402 goto out_dropped;
403 }
404 }
405 err = mlx5_cmd_check(dev, err, in, out);
406 if (err) {
407 mlx5_core_warn(dev, "func_id 0x%x, npages %d, err %d\n",
408 func_id, npages, err);
409 goto out_dropped;
410 }
411
412 func_type = func_id_to_type(dev, func_id, ec_function);
413 dev->priv.page_counters[func_type] += npages;
414 dev->priv.fw_pages += npages;
415
416 mlx5_core_dbg(dev, "npages %d, ec_function %d, func_id 0x%x, err %d\n",
417 npages, ec_function, func_id, err);
418
419 kvfree(in);
420 return 0;
421
422 out_dropped:
423 dev->priv.give_pages_dropped += npages;
424 out_4k:
425 for (i--; i >= 0; i--)
426 free_4k(dev, MLX5_GET64(manage_pages_in, in, pas[i]), function);
427 out_free:
428 kvfree(in);
429 if (notify_fail)
430 page_notify_fail(dev, func_id, ec_function);
431 return err;
432 }
433
release_all_pages(struct mlx5_core_dev * dev,u16 func_id,bool ec_function)434 static void release_all_pages(struct mlx5_core_dev *dev, u16 func_id,
435 bool ec_function)
436 {
437 u32 function = get_function(func_id, ec_function);
438 struct rb_root *root;
439 struct rb_node *p;
440 int npages = 0;
441 u16 func_type;
442
443 root = xa_load(&dev->priv.page_root_xa, function);
444 if (WARN_ON_ONCE(!root))
445 return;
446
447 p = rb_first(root);
448 while (p) {
449 struct fw_page *fwp = rb_entry(p, struct fw_page, rb_node);
450
451 p = rb_next(p);
452 npages += (MLX5_NUM_4K_IN_PAGE - fwp->free_count);
453 free_fwp(dev, fwp, fwp->free_count);
454 }
455
456 func_type = func_id_to_type(dev, func_id, ec_function);
457 dev->priv.page_counters[func_type] -= npages;
458 dev->priv.fw_pages -= npages;
459
460 mlx5_core_dbg(dev, "npages %d, ec_function %d, func_id 0x%x\n",
461 npages, ec_function, func_id);
462 }
463
fwp_fill_manage_pages_out(struct fw_page * fwp,u32 * out,u32 index,u32 npages)464 static u32 fwp_fill_manage_pages_out(struct fw_page *fwp, u32 *out, u32 index,
465 u32 npages)
466 {
467 u32 pages_set = 0;
468 unsigned int n;
469
470 for_each_clear_bit(n, &fwp->bitmask, MLX5_NUM_4K_IN_PAGE) {
471 MLX5_ARRAY_SET64(manage_pages_out, out, pas, index + pages_set,
472 fwp->addr + (n * MLX5_ADAPTER_PAGE_SIZE));
473 pages_set++;
474
475 if (!--npages)
476 break;
477 }
478
479 return pages_set;
480 }
481
reclaim_pages_cmd(struct mlx5_core_dev * dev,u32 * in,int in_size,u32 * out,int out_size)482 static int reclaim_pages_cmd(struct mlx5_core_dev *dev,
483 u32 *in, int in_size, u32 *out, int out_size)
484 {
485 struct rb_root *root;
486 struct fw_page *fwp;
487 struct rb_node *p;
488 bool ec_function;
489 u32 func_id;
490 u32 npages;
491 u32 i = 0;
492
493 if (!mlx5_cmd_is_down(dev))
494 return mlx5_cmd_do(dev, in, in_size, out, out_size);
495
496 /* No hard feelings, we want our pages back! */
497 npages = MLX5_GET(manage_pages_in, in, input_num_entries);
498 func_id = MLX5_GET(manage_pages_in, in, function_id);
499 ec_function = MLX5_GET(manage_pages_in, in, embedded_cpu_function);
500
501 root = xa_load(&dev->priv.page_root_xa, get_function(func_id, ec_function));
502 if (WARN_ON_ONCE(!root))
503 return -EEXIST;
504
505 p = rb_first(root);
506 while (p && i < npages) {
507 fwp = rb_entry(p, struct fw_page, rb_node);
508 p = rb_next(p);
509
510 i += fwp_fill_manage_pages_out(fwp, out, i, npages - i);
511 }
512
513 MLX5_SET(manage_pages_out, out, output_num_entries, i);
514 return 0;
515 }
516
reclaim_pages(struct mlx5_core_dev * dev,u16 func_id,int npages,int * nclaimed,bool event,bool ec_function)517 static int reclaim_pages(struct mlx5_core_dev *dev, u16 func_id, int npages,
518 int *nclaimed, bool event, bool ec_function)
519 {
520 u32 function = get_function(func_id, ec_function);
521 int outlen = MLX5_ST_SZ_BYTES(manage_pages_out);
522 u32 in[MLX5_ST_SZ_DW(manage_pages_in)] = {};
523 int num_claimed;
524 u16 func_type;
525 u32 *out;
526 int err;
527 int i;
528
529 if (nclaimed)
530 *nclaimed = 0;
531
532 outlen += npages * MLX5_FLD_SZ_BYTES(manage_pages_out, pas[0]);
533 out = kvzalloc(outlen, GFP_KERNEL);
534 if (!out)
535 return -ENOMEM;
536
537 MLX5_SET(manage_pages_in, in, opcode, MLX5_CMD_OP_MANAGE_PAGES);
538 MLX5_SET(manage_pages_in, in, op_mod, MLX5_PAGES_TAKE);
539 MLX5_SET(manage_pages_in, in, function_id, func_id);
540 MLX5_SET(manage_pages_in, in, input_num_entries, npages);
541 MLX5_SET(manage_pages_in, in, embedded_cpu_function, ec_function);
542
543 mlx5_core_dbg(dev, "func 0x%x, npages %d, outlen %d\n",
544 func_id, npages, outlen);
545 err = reclaim_pages_cmd(dev, in, sizeof(in), out, outlen);
546 if (err) {
547 npages = MLX5_GET(manage_pages_in, in, input_num_entries);
548 dev->priv.reclaim_pages_discard += npages;
549 }
550 /* if triggered by FW event and failed by FW then ignore */
551 if (event && err == -EREMOTEIO) {
552 err = 0;
553 goto out_free;
554 }
555
556 err = mlx5_cmd_check(dev, err, in, out);
557 if (err) {
558 mlx5_core_err(dev, "failed reclaiming pages: err %d\n", err);
559 goto out_free;
560 }
561
562 num_claimed = MLX5_GET(manage_pages_out, out, output_num_entries);
563 if (num_claimed > npages) {
564 mlx5_core_warn(dev, "fw returned %d, driver asked %d => corruption\n",
565 num_claimed, npages);
566 err = -EINVAL;
567 goto out_free;
568 }
569
570 for (i = 0; i < num_claimed; i++)
571 free_4k(dev, MLX5_GET64(manage_pages_out, out, pas[i]), function);
572
573 if (nclaimed)
574 *nclaimed = num_claimed;
575
576 func_type = func_id_to_type(dev, func_id, ec_function);
577 dev->priv.page_counters[func_type] -= num_claimed;
578 dev->priv.fw_pages -= num_claimed;
579
580 out_free:
581 kvfree(out);
582 return err;
583 }
584
pages_work_handler(struct work_struct * work)585 static void pages_work_handler(struct work_struct *work)
586 {
587 struct mlx5_pages_req *req = container_of(work, struct mlx5_pages_req, work);
588 struct mlx5_core_dev *dev = req->dev;
589 int err = 0;
590
591 if (req->release_all)
592 release_all_pages(dev, req->func_id, req->ec_function);
593 else if (req->npages < 0)
594 err = reclaim_pages(dev, req->func_id, -1 * req->npages, NULL,
595 true, req->ec_function);
596 else if (req->npages > 0)
597 err = give_pages(dev, req->func_id, req->npages, 1, req->ec_function);
598
599 if (err)
600 mlx5_core_warn(dev, "%s fail %d\n",
601 req->npages < 0 ? "reclaim" : "give", err);
602
603 kfree(req);
604 }
605
606 enum {
607 EC_FUNCTION_MASK = 0x8000,
608 RELEASE_ALL_PAGES_MASK = 0x4000,
609 };
610
req_pages_handler(struct notifier_block * nb,unsigned long type,void * data)611 static int req_pages_handler(struct notifier_block *nb,
612 unsigned long type, void *data)
613 {
614 struct mlx5_pages_req *req;
615 struct mlx5_core_dev *dev;
616 struct mlx5_priv *priv;
617 struct mlx5_eqe *eqe;
618 bool ec_function;
619 bool release_all;
620 u16 func_id;
621 s32 npages;
622
623 priv = mlx5_nb_cof(nb, struct mlx5_priv, pg_nb);
624 dev = container_of(priv, struct mlx5_core_dev, priv);
625 eqe = data;
626
627 func_id = be16_to_cpu(eqe->data.req_pages.func_id);
628 npages = be32_to_cpu(eqe->data.req_pages.num_pages);
629 ec_function = be16_to_cpu(eqe->data.req_pages.ec_function) & EC_FUNCTION_MASK;
630 release_all = be16_to_cpu(eqe->data.req_pages.ec_function) &
631 RELEASE_ALL_PAGES_MASK;
632 mlx5_core_dbg(dev, "page request for func 0x%x, npages %d, release_all %d\n",
633 func_id, npages, release_all);
634 req = kzalloc(sizeof(*req), GFP_ATOMIC);
635 if (!req) {
636 mlx5_core_warn(dev, "failed to allocate pages request\n");
637 return NOTIFY_DONE;
638 }
639
640 req->dev = dev;
641 req->func_id = func_id;
642 req->npages = npages;
643 req->ec_function = ec_function;
644 req->release_all = release_all;
645 INIT_WORK(&req->work, pages_work_handler);
646 queue_work(dev->priv.pg_wq, &req->work);
647 return NOTIFY_OK;
648 }
649
mlx5_satisfy_startup_pages(struct mlx5_core_dev * dev,int boot)650 int mlx5_satisfy_startup_pages(struct mlx5_core_dev *dev, int boot)
651 {
652 u16 func_id;
653 s32 npages;
654 int err;
655
656 err = mlx5_cmd_query_pages(dev, &func_id, &npages, boot);
657 if (err)
658 return err;
659
660 mlx5_core_dbg(dev, "requested %d %s pages for func_id 0x%x\n",
661 npages, boot ? "boot" : "init", func_id);
662
663 return give_pages(dev, func_id, npages, 0, mlx5_core_is_ecpf(dev));
664 }
665
666 enum {
667 MLX5_BLKS_FOR_RECLAIM_PAGES = 12
668 };
669
optimal_reclaimed_pages(void)670 static int optimal_reclaimed_pages(void)
671 {
672 struct mlx5_cmd_prot_block *block;
673 struct mlx5_cmd_layout *lay;
674 int ret;
675
676 ret = (sizeof(lay->out) + MLX5_BLKS_FOR_RECLAIM_PAGES * sizeof(block->data) -
677 MLX5_ST_SZ_BYTES(manage_pages_out)) /
678 MLX5_FLD_SZ_BYTES(manage_pages_out, pas[0]);
679
680 return ret;
681 }
682
mlx5_reclaim_root_pages(struct mlx5_core_dev * dev,struct rb_root * root,u32 function)683 static int mlx5_reclaim_root_pages(struct mlx5_core_dev *dev,
684 struct rb_root *root, u32 function)
685 {
686 u64 recl_pages_to_jiffies = msecs_to_jiffies(mlx5_tout_ms(dev, RECLAIM_PAGES));
687 unsigned long end = jiffies + recl_pages_to_jiffies;
688
689 while (!RB_EMPTY_ROOT(root)) {
690 u32 ec_function = mlx5_get_ec_function(function);
691 u32 function_id = mlx5_get_func_id(function);
692 int nclaimed;
693 int err;
694
695 err = reclaim_pages(dev, function_id, optimal_reclaimed_pages(),
696 &nclaimed, false, ec_function);
697 if (err) {
698 mlx5_core_warn(dev, "reclaim_pages err (%d) func_id=0x%x ec_func=0x%x\n",
699 err, function_id, ec_function);
700 return err;
701 }
702
703 if (nclaimed)
704 end = jiffies + recl_pages_to_jiffies;
705
706 if (time_after(jiffies, end)) {
707 mlx5_core_warn(dev, "FW did not return all pages. giving up...\n");
708 break;
709 }
710 }
711
712 return 0;
713 }
714
mlx5_reclaim_startup_pages(struct mlx5_core_dev * dev)715 int mlx5_reclaim_startup_pages(struct mlx5_core_dev *dev)
716 {
717 struct rb_root *root;
718 unsigned long id;
719 void *entry;
720
721 xa_for_each(&dev->priv.page_root_xa, id, entry) {
722 root = entry;
723 mlx5_reclaim_root_pages(dev, root, id);
724 xa_erase(&dev->priv.page_root_xa, id);
725 kfree(root);
726 }
727
728 WARN_ON(!xa_empty(&dev->priv.page_root_xa));
729
730 WARN(dev->priv.fw_pages,
731 "FW pages counter is %d after reclaiming all pages\n",
732 dev->priv.fw_pages);
733 WARN(dev->priv.page_counters[MLX5_VF],
734 "VFs FW pages counter is %d after reclaiming all pages\n",
735 dev->priv.page_counters[MLX5_VF]);
736 WARN(dev->priv.page_counters[MLX5_HOST_PF],
737 "External host PF FW pages counter is %d after reclaiming all pages\n",
738 dev->priv.page_counters[MLX5_HOST_PF]);
739 WARN(dev->priv.page_counters[MLX5_EC_VF],
740 "EC VFs FW pages counter is %d after reclaiming all pages\n",
741 dev->priv.page_counters[MLX5_EC_VF]);
742
743 return 0;
744 }
745
mlx5_pagealloc_init(struct mlx5_core_dev * dev)746 int mlx5_pagealloc_init(struct mlx5_core_dev *dev)
747 {
748 INIT_LIST_HEAD(&dev->priv.free_list);
749 dev->priv.pg_wq = create_singlethread_workqueue("mlx5_page_allocator");
750 if (!dev->priv.pg_wq)
751 return -ENOMEM;
752
753 xa_init(&dev->priv.page_root_xa);
754 mlx5_pages_debugfs_init(dev);
755
756 return 0;
757 }
758
mlx5_pagealloc_cleanup(struct mlx5_core_dev * dev)759 void mlx5_pagealloc_cleanup(struct mlx5_core_dev *dev)
760 {
761 mlx5_pages_debugfs_cleanup(dev);
762 xa_destroy(&dev->priv.page_root_xa);
763 destroy_workqueue(dev->priv.pg_wq);
764 }
765
mlx5_pagealloc_start(struct mlx5_core_dev * dev)766 void mlx5_pagealloc_start(struct mlx5_core_dev *dev)
767 {
768 MLX5_NB_INIT(&dev->priv.pg_nb, req_pages_handler, PAGE_REQUEST);
769 mlx5_eq_notifier_register(dev, &dev->priv.pg_nb);
770 }
771
mlx5_pagealloc_stop(struct mlx5_core_dev * dev)772 void mlx5_pagealloc_stop(struct mlx5_core_dev *dev)
773 {
774 mlx5_eq_notifier_unregister(dev, &dev->priv.pg_nb);
775 flush_workqueue(dev->priv.pg_wq);
776 }
777
mlx5_wait_for_pages(struct mlx5_core_dev * dev,int * pages)778 int mlx5_wait_for_pages(struct mlx5_core_dev *dev, int *pages)
779 {
780 u64 recl_vf_pages_to_jiffies = msecs_to_jiffies(mlx5_tout_ms(dev, RECLAIM_VFS_PAGES));
781 unsigned long end = jiffies + recl_vf_pages_to_jiffies;
782 int prev_pages = *pages;
783
784 /* In case of internal error we will free the pages manually later */
785 if (dev->state == MLX5_DEVICE_STATE_INTERNAL_ERROR) {
786 mlx5_core_warn(dev, "Skipping wait for vf pages stage");
787 return 0;
788 }
789
790 mlx5_core_dbg(dev, "Waiting for %d pages\n", prev_pages);
791 while (*pages) {
792 if (time_after(jiffies, end)) {
793 mlx5_core_warn(dev, "aborting while there are %d pending pages\n", *pages);
794 return -ETIMEDOUT;
795 }
796 if (*pages < prev_pages) {
797 end = jiffies + recl_vf_pages_to_jiffies;
798 prev_pages = *pages;
799 }
800 msleep(50);
801 }
802
803 mlx5_core_dbg(dev, "All pages received\n");
804 return 0;
805 }
806