blob: 643b484bdf5ae5151417763422e2438b8dfe6095 [file] [log] [blame]
Dave Barach8a7fb0c2016-07-08 14:44:23 -04001/*
Ed Warnickecb9cada2015-12-08 15:45:58 -07002 *------------------------------------------------------------------
Dave Barach8a7fb0c2016-07-08 14:44:23 -04003 * svmtool.c
Ed Warnickecb9cada2015-12-08 15:45:58 -07004 *
5 * Copyright (c) 2009 Cisco and/or its affiliates.
6 * Licensed under the Apache License, Version 2.0 (the "License");
7 * you may not use this file except in compliance with the License.
8 * You may obtain a copy of the License at:
9 *
10 * http://www.apache.org/licenses/LICENSE-2.0
11 *
12 * Unless required by applicable law or agreed to in writing, software
13 * distributed under the License is distributed on an "AS IS" BASIS,
14 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15 * See the License for the specific language governing permissions and
16 * limitations under the License.
17 *------------------------------------------------------------------
18 */
19
20#include <stdio.h>
21#include <stdlib.h>
22#include <sys/types.h>
23#include <sys/mman.h>
24#include <sys/stat.h>
25#include <netinet/in.h>
26#include <signal.h>
27#include <pthread.h>
28#include <unistd.h>
29#include <time.h>
30#include <fcntl.h>
31#include <string.h>
32#include <vppinfra/clib.h>
33#include <vppinfra/vec.h>
34#include <vppinfra/hash.h>
35#include <vppinfra/bitmap.h>
36#include <vppinfra/fifo.h>
37#include <vppinfra/time.h>
38#include <vppinfra/mheap.h>
39#include <vppinfra/heap.h>
40#include <vppinfra/pool.h>
41#include <vppinfra/format.h>
42
43#include "svm.h"
44
45
46
47/*
48 * format_all_svm_regions
49 * Maps / unmaps regions. Do NOT call from client code!
50 */
Dave Barach8a7fb0c2016-07-08 14:44:23 -040051u8 *
52format_all_svm_regions (u8 * s, va_list * args)
Ed Warnickecb9cada2015-12-08 15:45:58 -070053{
Dave Barach8a7fb0c2016-07-08 14:44:23 -040054 int verbose = va_arg (*args, int);
55 svm_region_t *root_rp = svm_get_root_rp ();
56 svm_main_region_t *mp;
57 svm_subregion_t *subp;
58 svm_region_t *rp;
59 svm_map_region_args_t *a = 0;
60 u8 **svm_names = 0;
61 u8 *name = 0;
62 int i;
Ed Warnickecb9cada2015-12-08 15:45:58 -070063
Dave Barach8a7fb0c2016-07-08 14:44:23 -040064 ASSERT (root_rp);
Ed Warnickecb9cada2015-12-08 15:45:58 -070065
Dave Barach8a7fb0c2016-07-08 14:44:23 -040066 pthread_mutex_lock (&root_rp->mutex);
Ed Warnickecb9cada2015-12-08 15:45:58 -070067
Dave Barach8a7fb0c2016-07-08 14:44:23 -040068 s = format (s, "%U", format_svm_region, root_rp, verbose);
Ed Warnickecb9cada2015-12-08 15:45:58 -070069
Dave Barach8a7fb0c2016-07-08 14:44:23 -040070 mp = root_rp->data_base;
Ed Warnickecb9cada2015-12-08 15:45:58 -070071
Dave Barach8a7fb0c2016-07-08 14:44:23 -040072 /*
73 * Snapshoot names, can't hold root rp mutex across
74 * find_or_create.
75 */
76 /* *INDENT-OFF* */
77 pool_foreach (subp, mp->subregions, ({
78 name = vec_dup (subp->subregion_name);
79 vec_add1(svm_names, name);
80 }));
81 /* *INDENT-ON* */
Ed Warnickecb9cada2015-12-08 15:45:58 -070082
Dave Barach8a7fb0c2016-07-08 14:44:23 -040083 pthread_mutex_unlock (&root_rp->mutex);
Ed Warnickecb9cada2015-12-08 15:45:58 -070084
Dave Barach8a7fb0c2016-07-08 14:44:23 -040085 for (i = 0; i < vec_len (svm_names); i++)
86 {
87 vec_validate (a, 0);
88 a->name = (char *) svm_names[i];
89 rp = svm_region_find_or_create (a);
90 if (rp)
91 {
92 pthread_mutex_lock (&rp->mutex);
93 s = format (s, "%U", format_svm_region, rp, verbose);
94 pthread_mutex_unlock (&rp->mutex);
95 svm_region_unmap (rp);
96 vec_free (svm_names[i]);
97 }
98 vec_free (a);
Ed Warnickecb9cada2015-12-08 15:45:58 -070099 }
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400100 vec_free (svm_names);
101 return (s);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700102}
103
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400104void
105show (char *chroot_path, int verbose)
Ed Warnickecb9cada2015-12-08 15:45:58 -0700106{
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400107 svm_map_region_args_t *a = 0;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700108
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400109 vec_validate (a, 0);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700110
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400111 svm_region_init_chroot (chroot_path);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700112
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400113 fformat (stdout, "My pid is %d\n", getpid ());
Ed Warnickecb9cada2015-12-08 15:45:58 -0700114
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400115 fformat (stdout, "%U", format_all_svm_regions, verbose);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700116
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400117 svm_region_exit ();
Ed Warnickecb9cada2015-12-08 15:45:58 -0700118
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400119 vec_free (a);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700120}
121
122
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400123static void *
124svm_map_region_nolock (svm_map_region_args_t * a)
Ed Warnickecb9cada2015-12-08 15:45:58 -0700125{
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400126 int svm_fd;
127 svm_region_t *rp;
128 int deadman = 0;
129 u8 *shm_name;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700130
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400131 ASSERT ((a->size & ~(MMAP_PAGESIZE - 1)) == a->size);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700132
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400133 shm_name = shm_name_from_svm_map_region_args (a);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700134
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400135 svm_fd = shm_open ((char *) shm_name, O_RDWR, 0777);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700136
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400137 if (svm_fd < 0)
138 {
139 perror ("svm_region_map(mmap open)");
140 return (0);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700141 }
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400142 vec_free (shm_name);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700143
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400144 rp = mmap (0, MMAP_PAGESIZE, PROT_READ | PROT_WRITE, MAP_SHARED, svm_fd, 0);
145
146 if (rp == (svm_region_t *) MAP_FAILED)
147 {
148 close (svm_fd);
149 clib_warning ("mmap");
150 return (0);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700151 }
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400152 /*
153 * We lost the footrace to create this region; make sure
154 * the winner has crossed the finish line.
155 */
156 while (rp->version == 0 && deadman++ < 5)
157 {
158 sleep (1);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700159 }
160
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400161 /*
162 * <bleep>-ed?
163 */
164 if (rp->version == 0)
165 {
166 clib_warning ("rp->version %d not %d", rp->version, SVM_VERSION);
Dave Barachac0798d2016-07-26 10:30:50 -0400167 munmap (rp, MMAP_PAGESIZE);
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400168 return (0);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700169 }
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400170 /* Remap now that the region has been placed */
171 a->baseva = rp->virtual_base;
172 a->size = rp->virtual_size;
173 munmap (rp, MMAP_PAGESIZE);
174
Damjan Marion7bee80c2017-04-26 15:32:12 +0200175 rp = (void *) mmap (uword_to_pointer (a->baseva, void *), a->size,
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400176 PROT_READ | PROT_WRITE,
177 MAP_SHARED | MAP_FIXED, svm_fd, 0);
178 if ((uword) rp == (uword) MAP_FAILED)
179 {
180 clib_unix_warning ("mmap");
181 return (0);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700182 }
183
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400184 if ((uword) rp != rp->virtual_base)
185 {
186 clib_warning ("mmap botch");
187 }
188
189 if (pthread_mutex_trylock (&rp->mutex))
190 {
191 clib_warning ("rp->mutex LOCKED by pid %d, tag %d, cleared...",
192 rp->mutex_owner_pid, rp->mutex_owner_tag);
Dave Barachb7b92992018-10-17 10:38:51 -0400193 clib_memset (&rp->mutex, 0, sizeof (rp->mutex));
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400194
195 }
196 else
197 {
198 clib_warning ("mutex OK...\n");
199 pthread_mutex_unlock (&rp->mutex);
200 }
201
202 return ((void *) rp);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700203}
204
205/*
206 * rnd_pagesize
207 * Round to a pagesize multiple, presumably 4k works
208 */
Dave Barachb3d93da2016-08-03 14:34:38 -0400209static u64
210rnd_pagesize (u64 size)
Ed Warnickecb9cada2015-12-08 15:45:58 -0700211{
Dave Barachb3d93da2016-08-03 14:34:38 -0400212 u64 rv;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700213
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400214 rv = (size + (MMAP_PAGESIZE - 1)) & ~(MMAP_PAGESIZE - 1);
215 return (rv);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700216}
217
218#define MUTEX_DEBUG
219
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400220always_inline void
221region_lock (svm_region_t * rp, int tag)
Ed Warnickecb9cada2015-12-08 15:45:58 -0700222{
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400223 pthread_mutex_lock (&rp->mutex);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700224#ifdef MUTEX_DEBUG
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400225 rp->mutex_owner_pid = getpid ();
226 rp->mutex_owner_tag = tag;
227#endif
Ed Warnickecb9cada2015-12-08 15:45:58 -0700228}
229
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400230always_inline void
231region_unlock (svm_region_t * rp)
Ed Warnickecb9cada2015-12-08 15:45:58 -0700232{
233#ifdef MUTEX_DEBUG
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400234 rp->mutex_owner_pid = 0;
235 rp->mutex_owner_tag = 0;
236#endif
237 pthread_mutex_unlock (&rp->mutex);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700238}
239
240
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400241static void *
242svm_existing_region_map_nolock (void *root_arg, svm_map_region_args_t * a)
Ed Warnickecb9cada2015-12-08 15:45:58 -0700243{
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400244 svm_region_t *root_rp = root_arg;
245 svm_main_region_t *mp;
246 svm_region_t *rp;
247 void *oldheap;
248 uword *p;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700249
Dave Barachc3799992016-08-15 11:12:27 -0400250 a->size += MMAP_PAGESIZE +
Dave Barachb3d93da2016-08-03 14:34:38 -0400251 (a->pvt_heap_size ? a->pvt_heap_size : SVM_PVT_MHEAP_SIZE);
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400252 a->size = rnd_pagesize (a->size);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700253
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400254 region_lock (root_rp, 4);
255 oldheap = svm_push_pvt_heap (root_rp);
256 mp = root_rp->data_base;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700257
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400258 ASSERT (mp);
259
260 p = hash_get_mem (mp->name_hash, a->name);
261
262 if (p)
263 {
264 rp = svm_map_region_nolock (a);
265 region_unlock (root_rp);
266 svm_pop_heap (oldheap);
267 return rp;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700268 }
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400269 return 0;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700270
271}
272
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400273static void
274trace (char *chroot_path, char *name, int enable_disable)
Ed Warnickecb9cada2015-12-08 15:45:58 -0700275{
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400276 svm_map_region_args_t *a = 0;
277 svm_region_t *db_rp;
278 void *oldheap;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700279
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400280 vec_validate (a, 0);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700281
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400282 svm_region_init_chroot (chroot_path);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700283
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400284 a->name = name;
285 a->size = 1 << 20;
286 a->flags = SVM_FLAGS_MHEAP;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700287
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400288 db_rp = svm_region_find_or_create (a);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700289
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400290 ASSERT (db_rp);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700291
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400292 region_lock (db_rp, 20);
293
294 oldheap = svm_push_data_heap (db_rp);
295
296 mheap_trace (db_rp->data_heap, enable_disable);
297
298 svm_pop_heap (oldheap);
299 region_unlock (db_rp);
300
301 svm_region_unmap ((void *) db_rp);
302 svm_region_exit ();
303 vec_free (a);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700304}
305
306
307
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400308static void
309subregion_repair (char *chroot_path)
Ed Warnickecb9cada2015-12-08 15:45:58 -0700310{
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400311 int i;
312 svm_main_region_t *mp;
313 svm_map_region_args_t a;
314 svm_region_t *root_rp;
315 svm_region_t *rp;
316 svm_subregion_t *subp;
317 u8 *name = 0;
318 u8 **svm_names = 0;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700319
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400320 svm_region_init_chroot (chroot_path);
321 root_rp = svm_get_root_rp ();
322
323 pthread_mutex_lock (&root_rp->mutex);
324
325 mp = root_rp->data_base;
326
327 /*
328 * Snapshoot names, can't hold root rp mutex across
329 * find_or_create.
330 */
331 /* *INDENT-OFF* */
332 pool_foreach (subp, mp->subregions, ({
333 name = vec_dup (subp->subregion_name);
334 vec_add1(svm_names, name);
335 }));
336 /* *INDENT-ON* */
337
338 pthread_mutex_unlock (&root_rp->mutex);
339
340 for (i = 0; i < vec_len (svm_names); i++)
341 {
Dave Barachb7b92992018-10-17 10:38:51 -0400342 clib_memset (&a, 0, sizeof (a));
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400343 a.root_path = chroot_path;
344 a.name = (char *) svm_names[i];
345 fformat (stdout, "Checking %s region...\n", a.name);
346 rp = svm_existing_region_map_nolock (root_rp, &a);
347 if (rp)
348 {
349 svm_region_unmap (rp);
350 vec_free (svm_names[i]);
351 }
Ed Warnickecb9cada2015-12-08 15:45:58 -0700352 }
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400353 vec_free (svm_names);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700354}
355
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400356void
357repair (char *chroot_path, int crash_root_region)
Ed Warnickecb9cada2015-12-08 15:45:58 -0700358{
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400359 svm_region_t *root_rp = 0;
360 svm_map_region_args_t *a = 0;
361 void *svm_map_region (svm_map_region_args_t * a);
362 int svm_fd;
363 u8 *shm_name;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700364
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400365 fformat (stdout, "our pid: %d\n", getpid ());
Ed Warnickecb9cada2015-12-08 15:45:58 -0700366
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400367 vec_validate (a, 0);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700368
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400369 a->root_path = chroot_path;
370 a->name = SVM_GLOBAL_REGION_NAME;
Damjan Marionaec8f892018-01-08 16:35:35 +0100371 a->baseva = svm_get_global_region_base_va ();
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400372 a->size = SVM_GLOBAL_REGION_SIZE;
373 a->flags = SVM_FLAGS_NODATA;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700374
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400375 shm_name = shm_name_from_svm_map_region_args (a);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700376
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400377 svm_fd = shm_open ((char *) shm_name, O_RDWR, 0777);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700378
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400379 if (svm_fd < 0)
380 {
381 perror ("svm_region_map(mmap open)");
382 goto out;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700383 }
384
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400385 vec_free (shm_name);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700386
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400387 root_rp = mmap (0, MMAP_PAGESIZE,
388 PROT_READ | PROT_WRITE, MAP_SHARED, svm_fd, 0);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700389
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400390 if (root_rp == (svm_region_t *) MAP_FAILED)
391 {
392 close (svm_fd);
393 clib_warning ("mmap");
394 goto out;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700395 }
Ed Warnickecb9cada2015-12-08 15:45:58 -0700396
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400397 /* Remap now that the region has been placed */
398 clib_warning ("remap to 0x%x", root_rp->virtual_base);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700399
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400400 a->baseva = root_rp->virtual_base;
401 a->size = root_rp->virtual_size;
402 munmap (root_rp, MMAP_PAGESIZE);
403
Damjan Marion7bee80c2017-04-26 15:32:12 +0200404 root_rp = (void *) mmap (uword_to_pointer (a->baseva, void *), a->size,
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400405 PROT_READ | PROT_WRITE,
406 MAP_SHARED | MAP_FIXED, svm_fd, 0);
407 if ((uword) root_rp == (uword) MAP_FAILED)
408 {
409 clib_unix_warning ("mmap");
410 goto out;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700411 }
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400412
413 close (svm_fd);
414
415 if ((uword) root_rp != root_rp->virtual_base)
416 {
417 clib_warning ("mmap botch");
418 goto out;
419 }
420
421 if (pthread_mutex_trylock (&root_rp->mutex))
422 {
423 clib_warning ("root_rp->mutex LOCKED by pid %d, tag %d, cleared...",
424 root_rp->mutex_owner_pid, root_rp->mutex_owner_tag);
Dave Barachb7b92992018-10-17 10:38:51 -0400425 clib_memset (&root_rp->mutex, 0, sizeof (root_rp->mutex));
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400426 goto out;
427 }
428 else
429 {
430 clib_warning ("root_rp->mutex OK...\n");
431 pthread_mutex_unlock (&root_rp->mutex);
432 }
433
434out:
435 vec_free (a);
436 /*
437 * Now that the root region is known to be OK,
438 * fix broken subregions
439 */
440 subregion_repair (chroot_path);
441
442 if (crash_root_region)
443 {
444 clib_warning ("Leaving root region locked on purpose...");
445 pthread_mutex_lock (&root_rp->mutex);
446 root_rp->mutex_owner_pid = getpid ();
447 root_rp->mutex_owner_tag = 99;
448 }
449 svm_region_exit ();
Ed Warnickecb9cada2015-12-08 15:45:58 -0700450}
451
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400452int
453main (int argc, char **argv)
Ed Warnickecb9cada2015-12-08 15:45:58 -0700454{
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400455 unformat_input_t input;
456 int parsed = 0;
457 char *name;
458 char *chroot_path = 0;
459 u8 *chroot_u8;
Ed Warnickecb9cada2015-12-08 15:45:58 -0700460
Guanghua Zhang60f16c22019-08-12 09:04:35 +0800461 clib_mem_init_thread_safe (0, 128 << 20);
462
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400463 unformat_init_command_line (&input, argv);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700464
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400465 while (unformat_check_input (&input) != UNFORMAT_END_OF_INPUT)
466 {
467 if (unformat (&input, "show-verbose"))
468 {
469 show (chroot_path, 1);
470 parsed++;
471 }
472 else if (unformat (&input, "show"))
473 {
474 show (chroot_path, 0);
475 parsed++;
476 }
477 else if (unformat (&input, "client-scan"))
478 {
479 svm_client_scan (chroot_path);
480 parsed++;
481 }
482 else if (unformat (&input, "repair"))
483 {
484 repair (chroot_path, 0 /* fix it */ );
485 parsed++;
486 }
487 else if (unformat (&input, "crash"))
488 {
489 repair (chroot_path, 1 /* crash it */ );
490 parsed++;
491 }
492 else if (unformat (&input, "trace-on %s", &name))
493 {
494 trace (chroot_path, name, 1);
495 parsed++;
496 }
497 else if (unformat (&input, "trace-off %s", &name))
498 {
499 trace (chroot_path, name, 0);
500 parsed++;
501 }
502 else if (unformat (&input, "chroot %s", &chroot_u8))
503 {
504 chroot_path = (char *) chroot_u8;
505 }
506 else
507 {
508 break;
509 }
Ed Warnickecb9cada2015-12-08 15:45:58 -0700510 }
511
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400512 unformat_free (&input);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700513
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400514 if (!parsed)
515 {
516 fformat (stdout,
517 "%s: show | show-verbose | client-scan | trace-on <region-name>\n",
518 argv[0]);
519 fformat (stdout, " trace-off <region-name>\n");
Ed Warnickecb9cada2015-12-08 15:45:58 -0700520 }
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400521 exit (0);
Ed Warnickecb9cada2015-12-08 15:45:58 -0700522}
Dave Barach8a7fb0c2016-07-08 14:44:23 -0400523
524/*
525 * fd.io coding-style-patch-verification: ON
526 *
527 * Local Variables:
528 * eval: (c-set-style "gnu")
529 * End:
530 */