blob: 37bf693f1821855266fd7907095c7254e2ab1bc1 [file] [log] [blame]
Damjan Marion68b4da62018-09-30 18:26:20 +02001/*
2 * Copyright (c) 2018 Cisco and/or its affiliates.
3 * Licensed under the Apache License, Version 2.0 (the "License");
4 * you may not use this file except in compliance with the License.
5 * You may obtain a copy of the License at:
6 *
7 * http://www.apache.org/licenses/LICENSE-2.0
8 *
9 * Unless required by applicable law or agreed to in writing, software
10 * distributed under the License is distributed on an "AS IS" BASIS,
11 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12 * See the License for the specific language governing permissions and
13 * limitations under the License.
14 */
15
16#include <unistd.h>
17#include <sys/types.h>
18#include <sys/mount.h>
19#include <sys/mman.h>
20#include <sys/fcntl.h>
21#include <sys/stat.h>
22#include <unistd.h>
23
24#include <vppinfra/linux/syscall.h>
25#include <vppinfra/linux/sysfs.h>
26#include <vlib/vlib.h>
27#include <vlib/physmem.h>
28#include <vlib/unix/unix.h>
29#include <vlib/pci/pci.h>
30#include <vlib/linux/vfio.h>
31
32clib_error_t *
33vlib_physmem_shared_map_create (vlib_main_t * vm, char *name, uword size,
Damjan Marion567e61d2018-10-24 17:08:26 +020034 u32 log2_page_sz, u32 numa_node,
35 u32 * map_index)
Damjan Marion68b4da62018-09-30 18:26:20 +020036{
37 clib_pmalloc_main_t *pm = vm->physmem_main.pmalloc_main;
38 vlib_physmem_main_t *vpm = &vm->physmem_main;
39 vlib_physmem_map_t *map;
40 clib_pmalloc_arena_t *a;
41 clib_error_t *error = 0;
42 void *va;
Damjan Marion8e8d3c82018-10-23 22:54:40 +020043 uword i;
Damjan Marion68b4da62018-09-30 18:26:20 +020044
Damjan Marion567e61d2018-10-24 17:08:26 +020045 va = clib_pmalloc_create_shared_arena (pm, name, size, log2_page_sz,
46 numa_node);
Damjan Marion68b4da62018-09-30 18:26:20 +020047
48 if (va == 0)
49 return clib_error_return (0, "%U", format_clib_error,
50 clib_pmalloc_last_error (pm));
51
52 a = clib_pmalloc_get_arena (pm, va);
53
54 pool_get (vpm->maps, map);
55 *map_index = map->index = map - vpm->maps;
56 map->base = va;
57 map->fd = a->fd;
Damjan Marion567e61d2018-10-24 17:08:26 +020058 map->n_pages = a->n_pages * a->subpages_per_page;
59 map->log2_page_size = a->log2_subpage_sz;
Damjan Marion68b4da62018-09-30 18:26:20 +020060
61 for (i = 0; i < a->n_pages; i++)
62 {
Damjan Marion567e61d2018-10-24 17:08:26 +020063 uword pa =
64 clib_pmalloc_get_pa (pm, (u8 *) va + (i << a->log2_subpage_sz));
Damjan Marion68b4da62018-09-30 18:26:20 +020065
66 /* maybe iova */
67 if (pa == 0)
68 pa = pointer_to_uword (va);
69
70 vec_add1 (map->page_table, pa);
71 }
72
73 return error;
74}
75
76vlib_physmem_map_t *
77vlib_physmem_get_map (vlib_main_t * vm, u32 index)
78{
79 vlib_physmem_main_t *vpm = &vm->physmem_main;
80 return pool_elt_at_index (vpm->maps, index);
81}
82
83clib_error_t *
84vlib_physmem_init (vlib_main_t * vm)
85{
86 vlib_physmem_main_t *vpm = &vm->physmem_main;
87 clib_error_t *error = 0;
88 u64 *pt = 0;
89 void *p;
90
91 /* check if pagemap is accessible */
92 pt = clib_mem_vm_get_paddr (&pt, min_log2 (sysconf (_SC_PAGESIZE)), 1);
Damjan Marion8e8d3c82018-10-23 22:54:40 +020093 if (pt && pt[0])
Damjan Marion68b4da62018-09-30 18:26:20 +020094 vpm->flags |= VLIB_PHYSMEM_MAIN_F_HAVE_PAGEMAP;
95 vec_free (pt);
96
97 if ((error = linux_vfio_init (vm)))
98 return error;
99
100 p = clib_mem_alloc_aligned (sizeof (clib_pmalloc_main_t),
101 CLIB_CACHE_LINE_BYTES);
102 memset (p, 0, sizeof (clib_pmalloc_main_t));
103 vpm->pmalloc_main = (clib_pmalloc_main_t *) p;
104 clib_pmalloc_init (vpm->pmalloc_main, 0);
105
106 return error;
107}
108
109static clib_error_t *
110show_physmem (vlib_main_t * vm,
111 unformat_input_t * input, vlib_cli_command_t * cmd)
112{
113 vlib_physmem_main_t *vpm = &vm->physmem_main;
114 unformat_input_t _line_input, *line_input = &_line_input;
Mohsin Kazmi6ec99c32018-11-07 16:55:18 +0100115 u32 verbose = 0, map = 0;
Damjan Marion68b4da62018-09-30 18:26:20 +0200116
117 if (unformat_user (input, unformat_line_input, line_input))
118 {
119 while (unformat_check_input (line_input) != UNFORMAT_END_OF_INPUT)
120 {
121 if (unformat (line_input, "verbose"))
122 verbose = 1;
123 else if (unformat (line_input, "v"))
124 verbose = 1;
125 else if (unformat (line_input, "detail"))
126 verbose = 2;
127 else if (unformat (line_input, "d"))
128 verbose = 2;
Mohsin Kazmi6ec99c32018-11-07 16:55:18 +0100129 else if (unformat (line_input, "map"))
130 map = 1;
Damjan Marion68b4da62018-09-30 18:26:20 +0200131 else
132 break;
133 }
134 unformat_free (line_input);
135 }
136
Mohsin Kazmi6ec99c32018-11-07 16:55:18 +0100137 if (map)
138 vlib_cli_output (vm, " %U", format_pmalloc_map, vpm->pmalloc_main);
139 else
140 vlib_cli_output (vm, " %U", format_pmalloc, vpm->pmalloc_main, verbose);
141
Damjan Marion68b4da62018-09-30 18:26:20 +0200142 return 0;
143}
144
145/* *INDENT-OFF* */
146VLIB_CLI_COMMAND (show_physmem_command, static) = {
147 .path = "show physmem",
Mohsin Kazmi6ec99c32018-11-07 16:55:18 +0100148 .short_help = "show physmem [verbose | detail | map]",
Damjan Marion68b4da62018-09-30 18:26:20 +0200149 .function = show_physmem,
150};
151/* *INDENT-ON* */
152
153/*
154 * fd.io coding-style-patch-verification: ON
155 *
156 * Local Variables:
157 * eval: (c-set-style "gnu")
158 * End:
159 */