Whamcloud - gitweb
LU-8915 lnet: migrate LNet selftest session handling to Netlink
[fs/lustre-release.git] / lustre / utils / portals.c
1 /*
2  * Copyright (c) 2003, 2010, Oracle and/or its affiliates. All rights reserved.
3  *
4  * Copyright (c) 2013, 2017, Intel Corporation.
5  *
6  *   This file is part of Lustre, https://wiki.whamcloud.com/
7  *
8  *   Portals is free software; you can redistribute it and/or
9  *   modify it under the terms of version 2 of the GNU General Public
10  *   License as published by the Free Software Foundation.
11  *
12  *   Portals is distributed in the hope that it will be useful,
13  *   but WITHOUT ANY WARRANTY; without even the implied warranty of
14  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
15  *   GNU General Public License for more details.
16  *
17  *   You should have received a copy of the GNU General Public License
18  *   along with Portals; if not, write to the Free Software
19  *   Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
20  *
21  */
22 #include <errno.h>
23 #include <getopt.h>
24 #include <inttypes.h>
25 #include <limits.h>
26 #ifdef HAVE_NETDB_H
27 # include <netdb.h>
28 #endif
29 #include <stdarg.h>
30 #include <stdio.h>
31 #include <stdlib.h>
32 #include <string.h>
33 #include <sys/ioctl.h>
34 #include <sys/time.h>
35 #include <time.h>
36 #include <linux/types.h>
37
38 #include <libcfs/util/ioctl.h>
39 #include <linux/lnet/libcfs_debug.h>
40 #include <linux/lnet/lnet-dlc.h>
41 #include <linux/lnet/lnetctl.h>
42 #include <linux/lnet/nidstr.h>
43 #include <linux/lnet/socklnd.h>
44 #include <lnetconfig/liblnetconfig.h>
45 #include <lustre/lustreapi.h>
46
47 unsigned int libcfs_debug;
48 unsigned int libcfs_printk = D_CANTMASK;
49
50 static bool  g_net_interactive;
51 static bool  g_net_set;
52 static __u32 g_net;
53
54 #define IOC_BUF_SIZE    8192
55 static char local_buf[IOC_BUF_SIZE];
56 static char *ioc_buf = local_buf;
57
58 /* Convert a string boolean to an int; "enable" -> 1 */
59 int
60 lnet_parse_bool (int *b, char *str)
61 {
62         if (!strcasecmp(str, "no") ||
63             !strcasecmp(str, "n") ||
64             !strcasecmp(str, "off") ||
65             !strcasecmp(str, "down") ||
66             !strcasecmp(str, "disable")) {
67                 *b = 0;
68
69                 return 0;
70         }
71
72         if (!strcasecmp(str, "yes") ||
73             !strcasecmp(str, "y") ||
74             !strcasecmp(str, "on") ||
75             !strcasecmp(str, "up") ||
76             !strcasecmp(str, "enable")) {
77                 *b = 1;
78
79                 return 0;
80         }
81
82         return -1;
83 }
84
85 int
86 lnet_parse_port(int *port, char *str)
87 {
88         char *end;
89
90         *port = strtol(str, &end, 0);
91
92         if (*end == 0 &&                        /* parsed whole string */
93             *port > 0 && *port < 65536)         /* minimal sanity check */
94                 return 0;
95
96         return -1;
97 }
98
99 #ifdef HAVE_GETHOSTBYNAME
100 static struct hostent *
101 ptl_gethostbyname(char *hname)
102 {
103         struct hostent *he;
104
105         he = gethostbyname(hname);
106         if (!he) {
107                 switch (h_errno) {
108                 case HOST_NOT_FOUND:
109                 case NO_ADDRESS:
110                         fprintf(stderr, "Unable to resolve hostname: %s\n",
111                                 hname);
112                         break;
113                 default:
114                         fprintf(stderr, "gethostbyname error for %s: %s\n",
115                                 hname, strerror(h_errno));
116                         break;
117                 }
118                 return NULL;
119         }
120         return he;
121 }
122 #endif
123
124 int
125 lnet_parse_ipquad(__u32 *ipaddrp, char *str)
126 {
127         int a, b, c, d;
128
129         if (sscanf(str, "%d.%d.%d.%d", &a, &b, &c, &d) == 4 &&
130             (a & ~0xff) == 0 && (b & ~0xff) == 0 &&
131             (c & ~0xff) == 0 && (d & ~0xff) == 0) {
132                 *ipaddrp = (a << 24) | (b << 16) | (c << 8) | d;
133
134                 return 0;
135         }
136
137         return -1;
138 }
139
140 int
141 lnet_parse_ipaddr(__u32 *ipaddrp, char *str)
142 {
143 #ifdef HAVE_GETHOSTBYNAME
144         struct hostent *he;
145 #endif
146
147         if (!strcmp(str, "_all_")) {
148                 *ipaddrp = 0;
149                 return 0;
150         }
151
152         if (lnet_parse_ipquad(ipaddrp, str) == 0)
153                 return 0;
154
155 #ifdef HAVE_GETHOSTBYNAME
156         if ((('a' <= str[0] && str[0] <= 'z') ||
157              ('A' <= str[0] && str[0] <= 'Z')) &&
158               (he = ptl_gethostbyname(str)) != NULL) {
159                 __u32 addr = *(__u32 *)he->h_addr;
160
161                 *ipaddrp = ntohl(addr);         /* HOST byte order */
162                 return 0;
163         }
164 #endif
165
166         return -1;
167 }
168
169 char *
170 ptl_ipaddr_2_str(__u32 ipaddr, char *str, size_t strsize, int lookup)
171 {
172 #ifdef HAVE_GETHOSTBYNAME
173         __u32 net_ip;
174         struct hostent *he;
175
176         if (lookup) {
177                 net_ip = htonl(ipaddr);
178                 he = gethostbyaddr(&net_ip, sizeof(net_ip), AF_INET);
179                 if (he) {
180                         snprintf(str, strsize, "%s", he->h_name);
181                         return str;
182                 }
183         }
184 #endif
185
186         sprintf(str, "%d.%d.%d.%d",
187                 (ipaddr >> 24) & 0xff, (ipaddr >> 16) & 0xff,
188                 (ipaddr >> 8) & 0xff, ipaddr & 0xff);
189         return str;
190 }
191
192 int
193 lnet_parse_time(time_t *t, char *str)
194 {
195         char *end;
196         int n;
197         struct tm tm;
198
199         *t = strtol(str, &end, 0);
200         if (*end == 0) /* parsed whole string */
201                 return 0;
202
203         memset(&tm, 0, sizeof(tm));
204         n = sscanf(str, "%d-%d-%d-%d:%d:%d",
205                    &tm.tm_year, &tm.tm_mon, &tm.tm_mday,
206                    &tm.tm_hour, &tm.tm_min, &tm.tm_sec);
207         if (n != 6)
208                 return -1;
209
210         tm.tm_mon--;                    /* convert to 0 == Jan */
211         tm.tm_year -= 1900;             /* y2k quirk */
212         tm.tm_isdst = -1;               /* dunno if it's daylight savings... */
213
214         *t = mktime(&tm);
215         if (*t == (time_t)-1)
216                 return -1;
217
218         return 0;
219 }
220
221 int
222 lnet_parse_nid(char *nid_str, struct lnet_process_id *id_ptr)
223 {
224         id_ptr->pid = LNET_PID_ANY;
225         id_ptr->nid = libcfs_str2nid(nid_str);
226         if (id_ptr->nid == LNET_NID_ANY) {
227                 fprintf(stderr, "Can't parse nid \"%s\"\n", nid_str);
228                 return -1;
229         }
230
231         return 0;
232 }
233
234 static int g_net_is_set(char *cmd)
235 {
236         if (g_net_set)
237                 return 1;
238
239         if (cmd) {
240                 char *net;
241
242                 if (g_net_interactive)
243                         net = "network";
244                 else
245                         net = "--net";
246
247                 fprintf(stderr,
248                         "You must run '%s <network>' command before '%s'\n",
249                         cmd, net);
250                 return 0;
251         }
252
253         return 0;
254 }
255
256 static int g_net_is_compatible(char *cmd, ...)
257 {
258         va_list ap;
259         int nal;
260
261         if (!g_net_is_set(cmd))
262                 return 0;
263
264         va_start(ap, cmd);
265
266         do {
267                 nal = va_arg(ap, int);
268                 if (nal == LNET_NETTYP(g_net)) {
269                         va_end(ap);
270                         return 1;
271                 }
272         } while (nal != 0);
273
274         va_end(ap);
275
276         if (cmd)
277                 fprintf(stderr, "Command %s not compatible with %s NAL\n",
278                         cmd, libcfs_lnd2str(LNET_NETTYP(g_net)));
279
280         return 0;
281 }
282
283 int ptl_initialize(int argc, char **argv)
284 {
285         if (argc > 1)
286                 g_net_interactive = true;
287
288         register_ioc_dev(LNET_DEV_ID, LNET_DEV_PATH);
289
290         return 0;
291 }
292
293 int jt_ptl_network(int argc, char **argv)
294 {
295         struct libcfs_ioctl_data data;
296         __u32 net = LNET_NET_ANY;
297         int rc;
298
299         if (argc != 2) {
300                 fprintf(stderr, "usage: %s <net>|up|down\n", argv[0]);
301                 return -1;
302         }
303
304         if (!strcmp(argv[1], "unconfigure") || !strcmp(argv[1], "down")) {
305                 LIBCFS_IOC_INIT(data);
306                 rc = l_ioctl(LNET_DEV_ID, IOC_LIBCFS_UNCONFIGURE, &data);
307
308                 if (rc == 0) {
309                         printf("LNET ready to unload\n");
310                         return 0;
311                 }
312
313                 if (errno == ENODEV) {
314                         printf("LNET is currently not loaded.");
315                         return 0;
316                 }
317
318                 if (errno == EBUSY)
319                         fprintf(stderr, "LNET busy\n");
320                 else
321                         fprintf(stderr, "LNET unconfigure error %d: %s\n",
322                                 errno, strerror(errno));
323                 return -1;
324         } else if (!strcmp(argv[1], "configure") || !strcmp(argv[1], "up")) {
325                 LIBCFS_IOC_INIT(data);
326                 rc = l_ioctl(LNET_DEV_ID, IOC_LIBCFS_CONFIGURE, &data);
327
328                 if (rc == 0) {
329                         printf("LNET configured\n");
330                         return 0;
331                 }
332
333                 fprintf(stderr, "LNET configure error %d: %s\n",
334                         errno, strerror(errno));
335                 return -1;
336         }
337
338         net = libcfs_str2net(argv[1]);
339         if (net == LNET_NET_ANY) {
340                 fprintf(stderr, "Can't parse net %s\n", argv[1]);
341                 return -1;
342         }
343
344         g_net_set = true;
345         g_net = net;
346         return 0;
347 }
348
349 #define IOC_LIBCFS_GET_NI       _IOWR('e', 50, IOCTL_LIBCFS_TYPE)
350
351 int
352 jt_ptl_list_nids(int argc, char **argv)
353 {
354         struct libcfs_ioctl_data data;
355         int all = 0, return_nid = 0;
356         yaml_emitter_t request;
357         yaml_parser_t reply;
358         yaml_event_t event;
359         struct nl_sock *sk;
360         bool done = false;
361         int rc = 0;
362         int count;
363
364         all = (argc == 2) && (strcmp(argv[1], "all") == 0);
365         /* Hack to pass back value */
366         return_nid = (argc == 2) && (argv[1][0] == 1);
367
368         if ((argc > 2) && !(all || return_nid)) {
369                 fprintf(stderr, "usage: %s [all]\n", argv[0]);
370                 return 0;
371         }
372
373         sk = nl_socket_alloc();
374         if (!sk)
375                 goto old_api;
376
377         /* Setup parser to receive Netlink packets */
378         rc = yaml_parser_initialize(&reply);
379         if (rc == 0) {
380                 yaml_parser_log_error(&reply, stderr, NULL);
381                 goto old_api;
382         }
383
384         rc = yaml_parser_set_input_netlink(&reply, sk, false);
385         if (rc == 0) {
386                 yaml_parser_log_error(&reply, stderr, NULL);
387                 yaml_parser_delete(&reply);
388                 goto old_api;
389         }
390
391         /* Create Netlink emitter to send request to kernel */
392         rc = yaml_emitter_initialize(&request);
393         if (rc == 0) {
394                 yaml_parser_log_error(&reply, stderr, NULL);
395                 yaml_parser_delete(&reply);
396                 goto old_api;
397         }
398
399         rc = yaml_emitter_set_output_netlink(&request, sk, LNET_GENL_NAME, 1,
400                                              LNET_CMD_NETS, NLM_F_DUMP);
401         if (rc == 0) {
402                 yaml_emitter_log_error(&request, stderr);
403                 yaml_emitter_delete(&request);
404                 yaml_parser_delete(&reply);
405                 goto old_api;
406         }
407
408         yaml_emitter_open(&request);
409         yaml_document_start_event_initialize(&event, NULL, NULL, NULL, 0);
410         rc = yaml_emitter_emit(&request, &event);
411         if (rc == 0)
412                 goto emitter_error;
413
414         yaml_mapping_start_event_initialize(&event, NULL,
415                                             (yaml_char_t *)YAML_MAP_TAG,
416                                             1, YAML_ANY_MAPPING_STYLE);
417         rc = yaml_emitter_emit(&request, &event);
418         if (rc == 0)
419                 goto emitter_error;
420
421         yaml_scalar_event_initialize(&event, NULL,
422                                      (yaml_char_t *)YAML_STR_TAG,
423                                      (yaml_char_t *)"net",
424                                      strlen("net"), 1, 0,
425                                      YAML_PLAIN_SCALAR_STYLE);
426         rc = yaml_emitter_emit(&request, &event);
427         if (rc == 0)
428                 goto emitter_error;
429
430         /* no net_id */
431         if (!g_net_set || g_net == LNET_NET_ANY) {
432                 yaml_scalar_event_initialize(&event, NULL,
433                                              (yaml_char_t *)YAML_STR_TAG,
434                                              (yaml_char_t *)"",
435                                              strlen(""), 1, 0,
436                                              YAML_PLAIN_SCALAR_STYLE);
437                 rc = yaml_emitter_emit(&request, &event);
438                 if (rc == 0)
439                         goto emitter_error;
440         } else {
441                 char *net_id = libcfs_net2str(g_net);
442
443                 yaml_sequence_start_event_initialize(&event, NULL,
444                                                      (yaml_char_t *)YAML_SEQ_TAG,
445                                                      1, YAML_ANY_SEQUENCE_STYLE);
446                 rc = yaml_emitter_emit(&request, &event);
447                 if (rc == 0)
448                         goto emitter_error;
449
450                 yaml_mapping_start_event_initialize(&event, NULL,
451                                                     (yaml_char_t *)YAML_MAP_TAG,
452                                                     1, YAML_ANY_MAPPING_STYLE);
453                 rc = yaml_emitter_emit(&request, &event);
454                 if (rc == 0)
455                         goto emitter_error;
456
457                 yaml_scalar_event_initialize(&event, NULL,
458                                              (yaml_char_t *)YAML_STR_TAG,
459                                              (yaml_char_t *)"net type",
460                                              strlen("net type"),
461                                              1, 0, YAML_PLAIN_SCALAR_STYLE);
462                 rc = yaml_emitter_emit(&request, &event);
463                 if (rc == 0)
464                         goto emitter_error;
465
466                 yaml_scalar_event_initialize(&event, NULL,
467                                              (yaml_char_t *)YAML_STR_TAG,
468                                              (yaml_char_t *)net_id,
469                                              strlen(net_id), 1, 0,
470                                              YAML_PLAIN_SCALAR_STYLE);
471                 rc = yaml_emitter_emit(&request, &event);
472                 if (rc == 0)
473                         goto emitter_error;
474
475                 yaml_mapping_end_event_initialize(&event);
476                 rc = yaml_emitter_emit(&request, &event);
477                 if (rc == 0)
478                         goto emitter_error;
479
480                 yaml_sequence_end_event_initialize(&event);
481                 rc = yaml_emitter_emit(&request, &event);
482                 if (rc == 0)
483                         goto emitter_error;
484         }
485         yaml_mapping_end_event_initialize(&event);
486         rc = yaml_emitter_emit(&request, &event);
487         if (rc == 0)
488                 goto emitter_error;
489
490         yaml_document_end_event_initialize(&event, 0);
491         rc = yaml_emitter_emit(&request, &event);
492         if (rc == 0)
493                 goto emitter_error;
494
495         rc = yaml_emitter_close(&request);
496 emitter_error:
497         if (rc == 0) {
498                 yaml_emitter_log_error(&request, stderr);
499                 rc = -EINVAL;
500         }
501         yaml_emitter_delete(&request);
502
503         while (!done) {
504                 rc = yaml_parser_parse(&reply, &event);
505                 if (rc == 0)
506                         break;
507
508                 if (event.type == YAML_SCALAR_EVENT &&
509                     strcmp((char *)event.data.scalar.value, "nid") == 0) {
510                         char *tmp;
511
512                         yaml_event_delete(&event);
513                         rc = yaml_parser_parse(&reply, &event);
514                         if (rc == 0) {
515                                 yaml_event_delete(&event);
516                                 break;
517                         }
518
519                         tmp = (char *)event.data.scalar.value;
520                         if (all || strcmp(tmp, "0@lo") != 0) {
521                                 printf("%s\n", tmp);
522                                 if (return_nid) {
523                                         *(__u64 *)(argv[1]) = libcfs_str2nid(tmp);
524                                         return_nid--;
525                                 }
526                         }
527                 }
528                 done = (event.type == YAML_STREAM_END_EVENT);
529                 yaml_event_delete(&event);
530         }
531
532         if (rc == 0)
533                 yaml_parser_log_error(&reply, stderr, NULL);
534         yaml_parser_delete(&reply);
535 old_api:
536         if (sk)
537                 nl_socket_free(sk);
538         if (rc == 1)
539                 return 0;
540
541         for (count = 0;; count++) {
542                 LIBCFS_IOC_INIT(data);
543                 data.ioc_count = count;
544                 rc = l_ioctl(LNET_DEV_ID, IOC_LIBCFS_GET_NI, &data);
545
546                 if (rc < 0) {
547                         if ((count > 0) && (errno == ENOENT))
548                                 /* We found them all */
549                                 break;
550                         fprintf(stderr, "IOC_LIBCFS_GET_NI error %d: %s\n",
551                                 errno, strerror(errno));
552                         return -1;
553                 }
554
555                 if (all || (data.ioc_nid != LNET_NID_LO_0)) {
556                         printf("%s\n", libcfs_nid2str(data.ioc_nid));
557                         if (return_nid) {
558                                 *(__u64 *)(argv[1]) = data.ioc_nid;
559                                 return_nid--;
560                         }
561                 }
562         }
563
564         return 0;
565 }
566
567 int
568 jt_ptl_which_nid(int argc, char **argv)
569 {
570         struct libcfs_ioctl_data data;
571         int best_dist = 0;
572         int best_order = 0;
573         lnet_nid_t   best_nid = LNET_NID_ANY;
574         int dist;
575         int order;
576         lnet_nid_t nid;
577         char *nidstr;
578         int rc;
579         int i;
580
581         if (argc < 2) {
582                 fprintf(stderr, "usage: %s NID [NID...]\n", argv[0]);
583                 return 0;
584         }
585
586         for (i = 1; i < argc; i++) {
587                 nidstr = argv[i];
588                 nid = libcfs_str2nid(nidstr);
589                 if (nid == LNET_NID_ANY) {
590                         fprintf(stderr, "Can't parse NID %s\n", nidstr);
591                         return -1;
592                 }
593
594                 LIBCFS_IOC_INIT(data);
595                 data.ioc_nid = nid;
596
597                 rc = l_ioctl(LNET_DEV_ID, IOC_LIBCFS_LNET_DIST, &data);
598                 if (rc != 0) {
599                         fprintf(stderr, "Can't get distance to %s: %s\n",
600                                 nidstr, strerror(errno));
601                         return -1;
602                 }
603
604                 dist = data.ioc_u32[0];
605                 order = data.ioc_u32[1];
606
607                 if (dist < 0) {
608                         if (dist == -EHOSTUNREACH)
609                                 continue;
610
611                         fprintf(stderr, "Unexpected distance to %s: %d\n",
612                                 nidstr, dist);
613                         return -1;
614                 }
615
616                 if (best_nid == LNET_NID_ANY ||
617                     dist < best_dist ||
618                     (dist == best_dist && order < best_order)) {
619                         best_dist = dist;
620                         best_order = order;
621                         best_nid = nid;
622                 }
623         }
624
625         if (best_nid == LNET_NID_ANY) {
626                 fprintf(stderr, "No reachable NID\n");
627                 return -1;
628         }
629
630         printf("%s\n", libcfs_nid2str(best_nid));
631         return 0;
632 }
633
634 int
635 jt_ptl_print_interfaces(int argc, char **argv)
636 {
637         struct libcfs_ioctl_data data;
638         char buffer[3][HOST_NAME_MAX + 1];
639         int index;
640         int rc;
641
642         if (!g_net_is_compatible(argv[0], SOCKLND, 0))
643                 return -1;
644
645         for (index = 0; ; index++) {
646                 LIBCFS_IOC_INIT(data);
647                 data.ioc_net   = g_net;
648                 data.ioc_count = index;
649
650                 rc = l_ioctl(LNET_DEV_ID, IOC_LIBCFS_GET_INTERFACE, &data);
651                 if (rc != 0)
652                         break;
653
654                 printf("%s: (%s/%s) npeer %d nroute %d\n",
655                        ptl_ipaddr_2_str(data.ioc_u32[0], buffer[2],
656                                         sizeof(buffer[2]), 1),
657                        ptl_ipaddr_2_str(data.ioc_u32[0], buffer[0],
658                                         sizeof(buffer[0]), 0),
659                        ptl_ipaddr_2_str(data.ioc_u32[1], buffer[1],
660                                         sizeof(buffer[1]), 0),
661                        data.ioc_u32[2], data.ioc_u32[3]);
662         }
663
664         if (index == 0) {
665                 if (errno == ENOENT) {
666                         printf("<no interfaces>\n");
667                 } else {
668                         fprintf(stderr,
669                                 "Error getting interfaces: %s: check dmesg.\n",
670                                 strerror(errno));
671                 }
672         }
673
674         return 0;
675 }
676
677 int
678 jt_ptl_add_interface(int argc, char **argv)
679 {
680         struct libcfs_ioctl_data data;
681         __u32 ipaddr;
682         int rc;
683         __u32 netmask = 0xffffff00;
684         int i;
685         int count;
686         char *end;
687
688         if (argc < 2 || argc > 3) {
689                 fprintf(stderr, "usage: %s ipaddr [netmask]\n", argv[0]);
690                 return 0;
691         }
692
693         if (!g_net_is_compatible(argv[0], SOCKLND, 0))
694                 return -1;
695
696         if (lnet_parse_ipaddr(&ipaddr, argv[1]) != 0) {
697                 fprintf(stderr, "Can't parse ip: %s\n", argv[1]);
698                 return -1;
699         }
700
701         if (argc > 2) {
702                 count = strtol(argv[2], &end, 0);
703                 if (count > 0 && count < 32 && *end == 0) {
704                         netmask = 0;
705                         for (i = count; i > 0; i--)
706                                 netmask = netmask | (1 << (32 - i));
707                 } else if (lnet_parse_ipquad(&netmask, argv[2]) != 0) {
708                         fprintf(stderr, "Can't parse netmask: %s\n", argv[2]);
709                         return -1;
710                 }
711         }
712
713         LIBCFS_IOC_INIT(data);
714         data.ioc_net    = g_net;
715         data.ioc_u32[0] = ipaddr;
716         data.ioc_u32[1] = netmask;
717
718         rc = l_ioctl(LNET_DEV_ID, IOC_LIBCFS_ADD_INTERFACE, &data);
719         if (rc != 0) {
720                 fprintf(stderr, "failed to add interface: %s\n",
721                         strerror(errno));
722                 return -1;
723         }
724
725         return 0;
726 }
727
728 int
729 jt_ptl_del_interface(int argc, char **argv)
730 {
731         struct libcfs_ioctl_data data;
732         int rc;
733         __u32 ipaddr = 0;
734
735         if (argc > 2) {
736                 fprintf(stderr, "usage: %s [ipaddr]\n", argv[0]);
737                 return 0;
738         }
739
740         if (!g_net_is_compatible(argv[0], SOCKLND, 0))
741                 return -1;
742
743         if (argc == 2 &&
744             lnet_parse_ipaddr(&ipaddr, argv[1]) != 0) {
745                 fprintf(stderr, "Can't parse ip: %s\n", argv[1]);
746                 return -1;
747         }
748
749         LIBCFS_IOC_INIT(data);
750         data.ioc_net    = g_net;
751         data.ioc_u32[0] = ipaddr;
752
753         rc = l_ioctl(LNET_DEV_ID, IOC_LIBCFS_DEL_INTERFACE, &data);
754         if (rc != 0) {
755                 fprintf(stderr, "failed to delete interface: %s\n",
756                         strerror(errno));
757                 return -1;
758         }
759
760         return 0;
761 }
762
763 int
764 jt_ptl_print_peers(int argc, char **argv)
765 {
766         struct libcfs_ioctl_data data;
767         struct lnet_process_id id;
768         char buffer[2][HOST_NAME_MAX + 1];
769         int index;
770         int rc;
771
772         if (!g_net_is_compatible(argv[0], SOCKLND, O2IBLND, GNILND,
773                                  PTL4LND, 0))
774                 return -1;
775
776         for (index = 0; ; index++) {
777                 LIBCFS_IOC_INIT(data);
778                 data.ioc_net     = g_net;
779                 data.ioc_count   = index;
780
781                 rc = l_ioctl(LNET_DEV_ID, IOC_LIBCFS_GET_PEER, &data);
782                 if (rc != 0)
783                         break;
784
785                 if (g_net_is_compatible(NULL, SOCKLND, 0)) {
786                         id.nid = data.ioc_nid;
787                         id.pid = data.ioc_u32[4];
788                         printf("%-20s [%d]%s->%s:%d #%d\n",
789                                libcfs_id2str(id),
790                                data.ioc_count, /* persistence */
791                                /* my ip */
792                                ptl_ipaddr_2_str(data.ioc_u32[2], buffer[0],
793                                                 sizeof(buffer[0]), 1),
794                                /* peer ip */
795                                ptl_ipaddr_2_str(data.ioc_u32[0], buffer[1],
796                                                 sizeof(buffer[1]), 1),
797                                data.ioc_u32[1], /* peer port */
798                                data.ioc_u32[3]); /* conn_count */
799                 } else if (g_net_is_compatible(NULL, GNILND, 0)) {
800                         int disconn = data.ioc_flags >> 16;
801                         char *state;
802
803                         if (disconn)
804                                 state = "D";
805                         else
806                                 state = data.ioc_flags & 0xffff ? "C" : "U";
807
808                         printf("%-20s (%d) %s [%d] %ju sq %d/%d tx %d/%d/%d\n",
809                                libcfs_nid2str(data.ioc_nid), /* peer nid */
810                                data.ioc_net, /* gemini device id */
811                                state, /* peer is Connecting, Up, or Down */
812                                data.ioc_count,   /* peer refcount */
813                                (uintmax_t)data.ioc_u64[0], /* peerstamp */
814                                data.ioc_u32[2], data.ioc_u32[3], /* tx and rx seq */
815                                /* fmaq, nfma, nrdma */
816                                data.ioc_u32[0], data.ioc_u32[1],
817                                data.ioc_u32[4]);
818                 } else {
819                         printf("%-20s [%d]\n",
820                                libcfs_nid2str(data.ioc_nid), data.ioc_count);
821                 }
822         }
823
824         if (index == 0) {
825                 if (errno == ENOENT) {
826                         printf("<no peers>\n");
827                 } else {
828                         fprintf(stderr,
829                                 "Error getting peer list: %s: check dmesg.\n",
830                                 strerror(errno));
831                 }
832         }
833         return 0;
834 }
835
836 int jt_ptl_add_peer(int argc, char **argv)
837 {
838         struct libcfs_ioctl_data data;
839         lnet_nid_t nid;
840         __u32 ip = 0;
841         int port = 0;
842         int rc;
843
844         if (!g_net_is_compatible(argv[0], SOCKLND, GNILND, 0))
845                 return -1;
846
847         if (argc != 4) {
848                 fprintf(stderr, "usage(tcp,gni): %s nid ipaddr port\n",
849                         argv[0]);
850                 return 0;
851         }
852
853         nid = libcfs_str2nid(argv[1]);
854         if (nid == LNET_NID_ANY) {
855                 fprintf(stderr, "Can't parse NID: %s\n", argv[1]);
856                 return -1;
857         }
858
859         if (lnet_parse_ipaddr(&ip, argv[2]) != 0) {
860                 fprintf(stderr, "Can't parse ip addr: %s\n", argv[2]);
861                 return -1;
862         }
863
864         if (lnet_parse_port(&port, argv[3]) != 0) {
865                 fprintf(stderr, "Can't parse port: %s\n", argv[3]);
866                 return -1;
867         }
868
869         LIBCFS_IOC_INIT(data);
870         data.ioc_net    = g_net;
871         data.ioc_nid    = nid;
872         data.ioc_u32[0] = ip;
873         data.ioc_u32[1] = port;
874
875         rc = l_ioctl(LNET_DEV_ID, IOC_LIBCFS_ADD_PEER, &data);
876         if (rc != 0) {
877                 fprintf(stderr, "failed to add peer: %s\n",
878                         strerror(errno));
879                 return -1;
880         }
881
882         return 0;
883 }
884
885 int
886 jt_ptl_del_peer(int argc, char **argv)
887 {
888         struct libcfs_ioctl_data data;
889         lnet_nid_t nid = LNET_NID_ANY;
890         lnet_pid_t pid = LNET_PID_ANY;
891         __u32 ip = 0;
892         int rc;
893
894         if (!g_net_is_compatible(argv[0], SOCKLND, O2IBLND, GNILND,
895                                  PTL4LND, 0))
896                 return -1;
897
898         if (g_net_is_compatible(NULL, SOCKLND, 0)) {
899                 if (argc > 3) {
900                         fprintf(stderr, "usage: %s [nid] [ipaddr]\n", argv[0]);
901                         return 0;
902                 }
903         } else if (argc > 2) {
904                 fprintf(stderr, "usage: %s [nid]\n", argv[0]);
905                 return 0;
906         }
907
908         if (argc > 1 && !libcfs_str2anynid(&nid, argv[1])) {
909                 fprintf(stderr, "Can't parse nid: %s\n", argv[1]);
910                 return -1;
911         }
912
913         if (g_net_is_compatible(NULL, SOCKLND, 0)) {
914                 if (argc > 2 && lnet_parse_ipaddr(&ip, argv[2]) != 0) {
915                         fprintf(stderr, "Can't parse ip addr: %s\n", argv[2]);
916                         return -1;
917                 }
918         }
919
920         LIBCFS_IOC_INIT(data);
921         data.ioc_net    = g_net;
922         data.ioc_nid    = nid;
923         data.ioc_u32[0] = ip;
924         data.ioc_u32[1] = pid;
925
926         rc = l_ioctl(LNET_DEV_ID, IOC_LIBCFS_DEL_PEER, &data);
927         if (rc != 0) {
928                 fprintf(stderr, "failed to remove peer: %s\n", strerror(errno));
929                 return -1;
930         }
931
932         return 0;
933 }
934
935 int
936 jt_ptl_print_connections(int argc, char **argv)
937 {
938         struct libcfs_ioctl_data data;
939         struct lnet_process_id id;
940         char buffer[2][HOST_NAME_MAX + 1];
941         int index;
942         int rc;
943
944         if (!g_net_is_compatible(argv[0], SOCKLND, O2IBLND, GNILND, 0))
945                 return -1;
946
947         for (index = 0; ; index++) {
948                 LIBCFS_IOC_INIT(data);
949                 data.ioc_net     = g_net;
950                 data.ioc_count   = index;
951
952                 rc = l_ioctl(LNET_DEV_ID, IOC_LIBCFS_GET_CONN, &data);
953                 if (rc != 0)
954                         break;
955
956                 if (g_net_is_compatible(NULL, SOCKLND, 0)) {
957                         id.nid = data.ioc_nid;
958                         id.pid = data.ioc_u32[6];
959                         printf("%-20s %s[%d]%s->%s:%d %d/%d %s\n",
960                                libcfs_id2str(id),
961                                (data.ioc_u32[3] == SOCKLND_CONN_ANY) ? "A" :
962                                (data.ioc_u32[3] == SOCKLND_CONN_CONTROL) ? "C" :
963                                (data.ioc_u32[3] == SOCKLND_CONN_BULK_IN) ? "I" :
964                          (data.ioc_u32[3] == SOCKLND_CONN_BULK_OUT) ? "O" : "?",
965                                data.ioc_u32[4], /* scheduler */
966                                /* local IP addr */
967                                ptl_ipaddr_2_str(data.ioc_u32[2], buffer[0],
968                                                 sizeof(buffer[0]), 1),
969                                /* remote IP addr */
970                                ptl_ipaddr_2_str(data.ioc_u32[0], buffer[1],
971                                                 sizeof(buffer[1]), 1),
972                                data.ioc_u32[1],         /* remote port */
973                                data.ioc_count, /* tx buffer size */
974                                data.ioc_u32[5], /* rx buffer size */
975                                data.ioc_flags ? "nagle" : "nonagle");
976                 } else if (g_net_is_compatible(NULL, O2IBLND, 0)) {
977                         printf("%s mtu %d\n",
978                                libcfs_nid2str(data.ioc_nid),
979                                data.ioc_u32[0]); /* path MTU */
980                 } else if (g_net_is_compatible(NULL, GNILND, 0)) {
981                         printf("%-20s [%d]\n",
982                                libcfs_nid2str(data.ioc_nid),
983                                data.ioc_u32[0] /* device id */);
984                 } else {
985                         printf("%s\n", libcfs_nid2str(data.ioc_nid));
986                 }
987         }
988
989         if (index == 0) {
990                 if (errno == ENOENT) {
991                         printf("<no connections>\n");
992                 } else {
993                         fprintf(stderr,
994                                 "Error getting connection list: %s: check dmesg.\n",
995                                 strerror(errno));
996                 }
997         }
998         return 0;
999 }
1000
1001 int jt_ptl_disconnect(int argc, char **argv)
1002 {
1003         struct libcfs_ioctl_data data;
1004         lnet_nid_t nid = LNET_NID_ANY;
1005         __u32 ipaddr = 0;
1006         int rc;
1007
1008         if (argc > 3) {
1009                 fprintf(stderr, "usage: %s [nid] [ipaddr]\n", argv[0]);
1010                 return 0;
1011         }
1012
1013         if (!g_net_is_compatible(NULL, SOCKLND, O2IBLND, GNILND, 0))
1014                 return 0;
1015
1016         if (argc >= 2 && !libcfs_str2anynid(&nid, argv[1])) {
1017                 fprintf(stderr, "Can't parse nid %s\n", argv[1]);
1018                 return -1;
1019         }
1020
1021         if (g_net_is_compatible(NULL, SOCKLND, 0) && argc >= 3 &&
1022             lnet_parse_ipaddr(&ipaddr, argv[2]) != 0) {
1023                 fprintf(stderr, "Can't parse ip addr %s\n", argv[2]);
1024                 return -1;
1025         }
1026
1027         LIBCFS_IOC_INIT(data);
1028         data.ioc_net     = g_net;
1029         data.ioc_nid     = nid;
1030         data.ioc_u32[0]  = ipaddr;
1031
1032         rc = l_ioctl(LNET_DEV_ID, IOC_LIBCFS_CLOSE_CONNECTION, &data);
1033         if (rc != 0) {
1034                 fprintf(stderr, "failed to remove connection: %s\n",
1035                         strerror(errno));
1036                 return -1;
1037         }
1038
1039         return 0;
1040 }
1041
1042 int jt_ptl_push_connection(int argc, char **argv)
1043 {
1044         struct libcfs_ioctl_data data;
1045         int rc;
1046         lnet_nid_t nid = LNET_NID_ANY;
1047
1048         if (argc > 2) {
1049                 fprintf(stderr, "usage: %s [nid]\n", argv[0]);
1050                 return 0;
1051         }
1052
1053         if (!g_net_is_compatible(argv[0], SOCKLND, GNILND, 0))
1054                 return -1;
1055
1056         if (argc > 1 && !libcfs_str2anynid(&nid, argv[1])) {
1057                 fprintf(stderr, "Can't parse nid: %s\n", argv[1]);
1058                 return -1;
1059         }
1060
1061         LIBCFS_IOC_INIT(data);
1062         data.ioc_net     = g_net;
1063         data.ioc_nid     = nid;
1064
1065         rc = l_ioctl(LNET_DEV_ID, IOC_LIBCFS_PUSH_CONNECTION, &data);
1066         if (rc != 0) {
1067                 fprintf(stderr, "failed to push connection: %s\n",
1068                         strerror(errno));
1069                 return -1;
1070         }
1071
1072         return 0;
1073 }
1074
1075 int jt_ptl_ping(int argc, char **argv)
1076 {
1077         int rc;
1078         int timeout;
1079         struct lnet_process_id id;
1080         struct lnet_process_id ids[16];
1081         int maxids = sizeof(ids) / sizeof(ids[0]);
1082         struct libcfs_ioctl_data data;
1083         char *sep;
1084         int i;
1085
1086         if (argc < 2) {
1087                 fprintf(stderr, "usage: %s id [timeout (secs)]\n", argv[0]);
1088                 return 0;
1089         }
1090
1091         sep = strchr(argv[1], '-');
1092         if (!sep) {
1093                 rc = lnet_parse_nid(argv[1], &id);
1094                 if (rc != 0)
1095                         return -1;
1096         } else {
1097                 char   *end;
1098
1099                 if (argv[1][0] == 'u' || argv[1][0] == 'U')
1100                         id.pid = strtoul(&argv[1][1], &end, 0) |
1101                                 LNET_PID_USERFLAG;
1102                 else
1103                         id.pid = strtoul(argv[1], &end, 0);
1104
1105                 if (end != sep) { /* assuming '-' is part of hostname */
1106                         rc = lnet_parse_nid(argv[1], &id);
1107                         if (rc != 0)
1108                                 return -1;
1109                 } else {
1110                         id.nid = libcfs_str2nid(sep + 1);
1111
1112                         if (id.nid == LNET_NID_ANY) {
1113                                 fprintf(stderr,
1114                                         "Can't parse process id \"%s\"\n",
1115                                         argv[1]);
1116                                 return -1;
1117                         }
1118                 }
1119         }
1120
1121         if (argc > 2) {
1122                 timeout = 1000 * atol(argv[2]);
1123                 if (timeout > 120 * 1000) {
1124                         fprintf(stderr, "Timeout %s is to large\n",
1125                                 argv[2]);
1126                         return -1;
1127                 }
1128         } else {
1129                 timeout = 1000; /* default 1 second timeout */
1130         }
1131
1132         LIBCFS_IOC_INIT(data);
1133         data.ioc_nid     = id.nid;
1134         data.ioc_u32[0]  = id.pid;
1135         data.ioc_u32[1]  = timeout;
1136         data.ioc_plen1   = sizeof(ids);
1137         data.ioc_pbuf1   = (char *)ids;
1138
1139         rc = l_ioctl(LNET_DEV_ID, IOC_LIBCFS_PING, &data);
1140         if (rc != 0) {
1141                 fprintf(stderr, "failed to ping %s: %s\n",
1142                         id.pid == LNET_PID_ANY ?
1143                         libcfs_nid2str(id.nid) : libcfs_id2str(id),
1144                         strerror(errno));
1145                 return -1;
1146         }
1147
1148         for (i = 0; i < data.ioc_count && i < maxids; i++)
1149                 printf("%s\n", libcfs_id2str(ids[i]));
1150
1151         if (data.ioc_count > maxids)
1152                 printf("%d out of %d ids listed\n", maxids, data.ioc_count);
1153
1154         return 0;
1155 }
1156
1157 int jt_ptl_mynid(int argc, char **argv)
1158 {
1159         struct libcfs_ioctl_data data;
1160         lnet_nid_t nid;
1161         int rc;
1162
1163         if (argc != 2) {
1164                 fprintf(stderr, "usage: %s NID\n", argv[0]);
1165                 return 0;
1166         }
1167
1168         nid = libcfs_str2nid(argv[1]);
1169         if (nid == LNET_NID_ANY) {
1170                 fprintf(stderr, "Can't parse NID '%s'\n", argv[1]);
1171                 return -1;
1172         }
1173
1174         LIBCFS_IOC_INIT(data);
1175         data.ioc_net = LNET_NIDNET(nid);
1176         data.ioc_nid = nid;
1177
1178         rc = l_ioctl(LNET_DEV_ID, IOC_LIBCFS_REGISTER_MYNID, &data);
1179         if (rc < 0)
1180                 fprintf(stderr, "setting my NID failed: %s\n",
1181                         strerror(errno));
1182         else
1183                 printf("registered my nid %s\n", libcfs_nid2str(nid));
1184
1185         return 0;
1186 }
1187
1188 int
1189 jt_ptl_fail_nid(int argc, char **argv)
1190 {
1191         int rc;
1192         lnet_nid_t nid;
1193         int threshold;
1194         struct libcfs_ioctl_data data;
1195
1196         if (argc < 2 || argc > 3) {
1197                 fprintf(stderr, "usage: %s nid|\"*\" [count (0 == mend)]\n",
1198                         argv[0]);
1199                 return 0;
1200         }
1201
1202         if (!libcfs_str2anynid(&nid, argv[1])) {
1203                 fprintf(stderr, "Can't parse nid \"%s\"\n", argv[1]);
1204                 return -1;
1205         }
1206
1207         if (argc < 3) {
1208                 threshold = LNET_MD_THRESH_INF;
1209         } else if (sscanf(argv[2], "%i", &threshold) != 1) {
1210                 fprintf(stderr, "Can't parse count \"%s\"\n", argv[2]);
1211                 return -1;
1212         }
1213
1214         LIBCFS_IOC_INIT(data);
1215         data.ioc_nid = nid;
1216         data.ioc_count = threshold;
1217
1218         rc = l_ioctl(LNET_DEV_ID, IOC_LIBCFS_FAIL_NID, &data);
1219         if (rc < 0)
1220                 fprintf(stderr, "IOC_LIBCFS_FAIL_NID failed: %s\n",
1221                         strerror(errno));
1222         else
1223                 printf("%s %s\n",
1224                        threshold == 0 ? "Unfailing" : "Failing", argv[1]);
1225
1226         return 0;
1227 }
1228
1229 int
1230 jt_ptl_add_route(int argc, char **argv)
1231 {
1232         struct lnet_ioctl_config_data data;
1233         lnet_nid_t gateway_nid;
1234         __u32 hops = LNET_UNDEFINED_HOPS;
1235         unsigned int priority = 0;
1236         char *end;
1237         int rc;
1238
1239         if (argc < 2 || argc > 4) {
1240                 fprintf(stderr, "usage: %s gateway [hopcount [priority]]\n",
1241                         argv[0]);
1242                 return -1;
1243         }
1244
1245         if (g_net_is_set(argv[0]) == 0)
1246                 return -1;
1247
1248         gateway_nid = libcfs_str2nid(argv[1]);
1249         if (gateway_nid == LNET_NID_ANY) {
1250                 fprintf(stderr, "Can't parse gateway NID \"%s\"\n", argv[1]);
1251                 return -1;
1252         }
1253
1254         if (argc > 2) {
1255                 hops = strtol(argv[2], &end, 0);
1256                 if (hops == 0 || hops >= 256 ||
1257                     (end && *end != 0)) {
1258                         fprintf(stderr, "Can't parse hopcount \"%s\"\n",
1259                                 argv[2]);
1260                         return -1;
1261                 }
1262                 if (argc == 4) {
1263                         priority = strtoul(argv[3], &end, 0);
1264                         if (end && *end != 0) {
1265                                 fprintf(stderr,
1266                                         "Can't parse priority \"%s\"\n",
1267                                         argv[3]);
1268                                 return -1;
1269                         }
1270                 }
1271         }
1272
1273         LIBCFS_IOC_INIT_V2(data, cfg_hdr);
1274         data.cfg_net = g_net;
1275         data.cfg_config_u.cfg_route.rtr_hop = hops;
1276         data.cfg_nid = gateway_nid;
1277         data.cfg_config_u.cfg_route.rtr_priority = priority;
1278
1279         rc = l_ioctl(LNET_DEV_ID, IOC_LIBCFS_ADD_ROUTE, &data);
1280         if (rc != 0) {
1281                 fprintf(stderr, "IOC_LIBCFS_ADD_ROUTE failed: %s\n",
1282                         strerror(errno));
1283                 return -1;
1284         }
1285
1286         return 0;
1287 }
1288
1289 int
1290 jt_ptl_del_route(int argc, char **argv)
1291 {
1292         struct lnet_ioctl_config_data data;
1293         lnet_nid_t nid;
1294         int rc;
1295
1296         if (argc != 2) {
1297                 fprintf(stderr, "usage: %s gatewayNID\n", argv[0]);
1298                 return 0;
1299         }
1300
1301         if (libcfs_str2anynid(&nid, argv[1]) == 0) {
1302                 fprintf(stderr, "Can't parse gateway NID \"%s\"\n", argv[1]);
1303                 return -1;
1304         }
1305
1306         LIBCFS_IOC_INIT_V2(data, cfg_hdr);
1307         data.cfg_net = g_net_set ? g_net : LNET_NET_ANY;
1308         data.cfg_nid = nid;
1309
1310         rc = l_ioctl(LNET_DEV_ID, IOC_LIBCFS_DEL_ROUTE, &data);
1311         if (rc != 0) {
1312                 fprintf(stderr, "IOC_LIBCFS_DEL_ROUTE (%s) failed: %s\n",
1313                         libcfs_nid2str(nid), strerror(errno));
1314                 return -1;
1315         }
1316
1317         return 0;
1318 }
1319
1320 int
1321 jt_ptl_notify_router(int argc, char **argv)
1322 {
1323         struct libcfs_ioctl_data data;
1324         int enable;
1325         lnet_nid_t nid;
1326         int rc;
1327         struct timeval now;
1328         time_t when;
1329
1330         if (argc < 3) {
1331                 fprintf(stderr, "usage: %s targetNID <up/down> [<time>]\n",
1332                         argv[0]);
1333                 return 0;
1334         }
1335
1336         nid = libcfs_str2nid(argv[1]);
1337         if (nid == LNET_NID_ANY) {
1338                 fprintf(stderr, "Can't parse target NID \"%s\"\n", argv[1]);
1339                 return -1;
1340         }
1341
1342         if (lnet_parse_bool (&enable, argv[2]) != 0) {
1343                 fprintf(stderr, "Can't parse boolean %s\n", argv[2]);
1344                 return -1;
1345         }
1346
1347         gettimeofday(&now, NULL);
1348
1349         if (argc < 4) {
1350                 when = now.tv_sec;
1351         } else if (lnet_parse_time(&when, argv[3]) != 0) {
1352                 fprintf(stderr,
1353                         "Can't parse time %s\n Please specify either 'YYYY-MM-DD-HH:MM:SS'\n or an absolute unix time in seconds\n",
1354                         argv[3]);
1355                 return -1;
1356         } else if (when > now.tv_sec) {
1357                 fprintf(stderr, "%s specifies a time in the future\n",
1358                         argv[3]);
1359                 return -1;
1360         }
1361
1362         LIBCFS_IOC_INIT(data);
1363         data.ioc_nid = nid;
1364         data.ioc_flags = enable;
1365         /* Yeuch; 'cept I need a __u64 on 64 bit machines... */
1366         data.ioc_u64[0] = (__u64)when;
1367
1368         rc = l_ioctl(LNET_DEV_ID, IOC_LIBCFS_NOTIFY_ROUTER, &data);
1369         if (rc != 0) {
1370                 fprintf(stderr, "IOC_LIBCFS_NOTIFY_ROUTER (%s) failed: %s\n",
1371                         libcfs_nid2str(nid), strerror(errno));
1372                 return -1;
1373         }
1374
1375         return 0;
1376 }
1377
1378 int
1379 jt_ptl_print_routes(int argc, char **argv)
1380 {
1381         struct lnet_ioctl_config_data  data;
1382         int rc;
1383         int index;
1384         __u32 net;
1385         lnet_nid_t nid;
1386         unsigned int hops;
1387         int alive;
1388         unsigned int pri;
1389
1390         for (index = 0; ; index++) {
1391                 LIBCFS_IOC_INIT_V2(data, cfg_hdr);
1392                 data.cfg_count = index;
1393
1394                 rc = l_ioctl(LNET_DEV_ID, IOC_LIBCFS_GET_ROUTE, &data);
1395                 if (rc != 0)
1396                         break;
1397
1398                 net     = data.cfg_net;
1399                 hops    = data.cfg_config_u.cfg_route.rtr_hop;
1400                 nid     = data.cfg_nid;
1401                 alive   = data.cfg_config_u.cfg_route.rtr_flags & LNET_RT_ALIVE;
1402                 pri     = data.cfg_config_u.cfg_route.rtr_priority;
1403
1404                 printf("net %18s hops %u gw %32s %s pri %u\n",
1405                        libcfs_net2str(net), hops,
1406                        libcfs_nid2str(nid), alive ? "up" : "down", pri);
1407         }
1408
1409         if (errno != ENOENT)
1410                 fprintf(stderr, "Error getting routes: %s: check dmesg.\n",
1411                         strerror(errno));
1412
1413         return 0;
1414 }
1415
1416 static int
1417 fault_attr_nid_parse(char *str, lnet_nid_t *nid_p)
1418 {
1419         lnet_nid_t nid;
1420         __u32 net;
1421         int rc = 0;
1422
1423         /* NB: can't support range ipaddress except * and *@net */
1424         if (strlen(str) > 2 && str[0] == '*' && str[1] == '@') {
1425                 net = libcfs_str2net(str + 2);
1426                 if (net == LNET_NET_ANY)
1427                         goto failed;
1428
1429                 nid = LNET_MKNID(net, LNET_NIDADDR(LNET_NID_ANY));
1430         } else {
1431                 rc = libcfs_str2anynid(&nid, str);
1432                 if (!rc)
1433                         goto failed;
1434         }
1435
1436         *nid_p = nid;
1437         return 0;
1438 failed:
1439         fprintf(stderr, "Invalid NID : %s\n", str);
1440         return -1;
1441 }
1442
1443 static int
1444 fault_attr_msg_parse(char *msg_str, __u32 *mask_p)
1445 {
1446         if (!strcasecmp(msg_str, "put")) {
1447                 *mask_p |= LNET_PUT_BIT;
1448                 return 0;
1449
1450         } else if (!strcasecmp(msg_str, "ack")) {
1451                 *mask_p |= LNET_ACK_BIT;
1452                 return 0;
1453
1454         } else if (!strcasecmp(msg_str, "get")) {
1455                 *mask_p |= LNET_GET_BIT;
1456                 return 0;
1457
1458         } else if (!strcasecmp(msg_str, "reply")) {
1459                 *mask_p |= LNET_REPLY_BIT;
1460                 return 0;
1461         }
1462
1463         fprintf(stderr, "unknown message type %s\n", msg_str);
1464         return -1;
1465 }
1466
1467 static int
1468 fault_attr_ptl_parse(char *ptl_str, __u64 *mask_p)
1469 {
1470         unsigned long rc = strtoul(optarg, NULL, 0);
1471
1472         if (rc >= 64) {
1473                 fprintf(stderr, "invalid portal: %lu\n", rc);
1474                 return -1;
1475         }
1476
1477         *mask_p |= (1ULL << rc);
1478         return 0;
1479 }
1480
1481 static int
1482 fault_attr_health_error_parse(char *error, __u32 *mask)
1483 {
1484         if (!strcasecmp(error, "local_interrupt")) {
1485                 *mask |= HSTATUS_LOCAL_INTERRUPT_BIT;
1486                 return 0;
1487         }
1488         if (!strcasecmp(error, "local_dropped")) {
1489                 *mask |= HSTATUS_LOCAL_DROPPED_BIT;
1490                 return 0;
1491         }
1492         if (!strcasecmp(error, "local_aborted")) {
1493                 *mask |= HSTATUS_LOCAL_ABORTED_BIT;
1494                 return 0;
1495         }
1496         if (!strcasecmp(error, "local_no_route")) {
1497                 *mask |= HSTATUS_LOCAL_NO_ROUTE_BIT;
1498                 return 0;
1499         }
1500         if (!strcasecmp(error, "local_error")) {
1501                 *mask |= HSTATUS_LOCAL_ERROR_BIT;
1502                 return 0;
1503         }
1504         if (!strcasecmp(error, "local_timeout")) {
1505                 *mask |= HSTATUS_LOCAL_TIMEOUT_BIT;
1506                 return 0;
1507         }
1508         if (!strcasecmp(error, "remote_error")) {
1509                 *mask |= HSTATUS_REMOTE_ERROR_BIT;
1510                 return 0;
1511         }
1512         if (!strcasecmp(error, "remote_dropped")) {
1513                 *mask |= HSTATUS_REMOTE_DROPPED_BIT;
1514                 return 0;
1515         }
1516         if (!strcasecmp(error, "remote_timeout")) {
1517                 *mask |= HSTATUS_REMOTE_TIMEOUT_BIT;
1518                 return 0;
1519         }
1520         if (!strcasecmp(error, "network_timeout")) {
1521                 *mask |= HSTATUS_NETWORK_TIMEOUT_BIT;
1522                 return 0;
1523         }
1524         if (!strcasecmp(error, "random")) {
1525                 *mask = HSTATUS_RANDOM;
1526                 return 0;
1527         }
1528
1529         return -1;
1530 }
1531
1532 static int
1533 fault_simul_rule_add(__u32 opc, char *name, int argc, char **argv)
1534 {
1535         struct libcfs_ioctl_data  data = { { 0 } };
1536         struct lnet_fault_attr    attr;
1537         char *optstr;
1538         int rc;
1539
1540         static const struct option opts[] = {
1541         { .name = "source",   .has_arg = required_argument, .val = 's' },
1542         { .name = "dest",     .has_arg = required_argument, .val = 'd' },
1543         { .name = "rate",     .has_arg = required_argument, .val = 'r' },
1544         { .name = "interval", .has_arg = required_argument, .val = 'i' },
1545         { .name = "random",   .has_arg = no_argument,       .val = 'n' },
1546         { .name = "latency",  .has_arg = required_argument, .val = 'l' },
1547         { .name = "portal",   .has_arg = required_argument, .val = 'p' },
1548         { .name = "message",  .has_arg = required_argument, .val = 'm' },
1549         { .name = "health_error",  .has_arg = required_argument, .val = 'e' },
1550         { .name = "local_nid",  .has_arg = required_argument, .val = 'o' },
1551         { .name = "drop_all",  .has_arg = no_argument, .val = 'x' },
1552         { .name = NULL } };
1553
1554         if (argc == 1) {
1555                 fprintf(stderr,
1556                         "Failed, please provide source, destination and rate of rule\n");
1557                 return -1;
1558         }
1559
1560         optstr = opc == LNET_CTL_DROP_ADD ? "s:d:o:r:i:p:m:e:nx" : "s:d:o:r:l:p:m:";
1561         memset(&attr, 0, sizeof(attr));
1562         while (1) {
1563                 int c = getopt_long(argc, argv, optstr, opts, NULL);
1564
1565                 if (c == -1)
1566                         break;
1567
1568                 switch (c) {
1569                 case 'o':
1570                         rc = fault_attr_nid_parse(optarg, &attr.fa_local_nid);
1571                         if (rc != 0)
1572                                 goto getopt_failed;
1573                         break;
1574                 case 's': /* source NID/NET */
1575                         rc = fault_attr_nid_parse(optarg, &attr.fa_src);
1576                         if (rc != 0)
1577                                 goto getopt_failed;
1578                         break;
1579
1580                 case 'd': /* dest NID/NET */
1581                         rc = fault_attr_nid_parse(optarg, &attr.fa_dst);
1582                         if (rc != 0)
1583                                 goto getopt_failed;
1584                         break;
1585
1586                 case 'r': /* drop rate */
1587                         if (opc == LNET_CTL_DROP_ADD)
1588                                 attr.u.drop.da_rate = strtoul(optarg, NULL, 0);
1589                         else
1590                                 attr.u.delay.la_rate = strtoul(optarg, NULL, 0);
1591                         break;
1592
1593                 case 'e':
1594                         if (opc == LNET_CTL_DROP_ADD) {
1595                                 rc = fault_attr_health_error_parse(optarg,
1596                                                                    &attr.u.drop.da_health_error_mask);
1597                                 if (rc)
1598                                         goto getopt_failed;
1599                         }
1600                         break;
1601
1602                 case 'x':
1603                         if (opc == LNET_CTL_DROP_ADD)
1604                                 attr.u.drop.da_drop_all = true;
1605                         break;
1606
1607                 case 'n':
1608                         if (opc == LNET_CTL_DROP_ADD)
1609                                 attr.u.drop.da_random = true;
1610                         break;
1611
1612                 case 'i': /* time interval (# seconds) for message drop */
1613                         if (opc == LNET_CTL_DROP_ADD)
1614                                 attr.u.drop.da_interval = strtoul(optarg,
1615                                                                   NULL, 0);
1616                         else
1617                                 attr.u.delay.la_interval = strtoul(optarg,
1618                                                                    NULL, 0);
1619                         break;
1620
1621                 case 'l': /* seconds to wait before activating rule */
1622                         attr.u.delay.la_latency = strtoul(optarg, NULL, 0);
1623                         break;
1624
1625                 case 'p': /* portal to filter */
1626                         rc = fault_attr_ptl_parse(optarg, &attr.fa_ptl_mask);
1627                         if (rc != 0)
1628                                 goto getopt_failed;
1629                         break;
1630
1631                 case 'm': /* message types to filter */
1632                         rc = fault_attr_msg_parse(optarg, &attr.fa_msg_mask);
1633                         if (rc != 0)
1634                                 goto getopt_failed;
1635                         break;
1636
1637                 default:
1638                         fprintf(stderr, "error: %s: option '%s' unrecognized\n",
1639                                 argv[0], argv[optind - 1]);
1640                         goto getopt_failed;
1641                 }
1642         }
1643         optind = 1;
1644
1645         if (opc == LNET_CTL_DROP_ADD) {
1646                 /* NB: drop rate and interval are exclusive to each other */
1647                 if (!((attr.u.drop.da_rate == 0) ^
1648                       (attr.u.drop.da_interval == 0))) {
1649                         fprintf(stderr,
1650                                 "please provide either drop rate or interval but not both at the same time.\n");
1651                         return -1;
1652                 }
1653
1654                 if (attr.u.drop.da_random &&
1655                     attr.u.drop.da_interval == 0) {
1656                         fprintf(stderr,
1657                                 "please provide an interval to randomize\n");
1658                         return -1;
1659                 }
1660         } else if (opc == LNET_CTL_DELAY_ADD) {
1661                 if (!((attr.u.delay.la_rate == 0) ^
1662                       (attr.u.delay.la_interval == 0))) {
1663                         fprintf(stderr,
1664                                 "please provide either delay rate or interval but not both at the same time.\n");
1665                         return -1;
1666                 }
1667
1668                 if (attr.u.delay.la_latency == 0) {
1669                         fprintf(stderr, "latency cannot be zero\n");
1670                         return -1;
1671                 }
1672         }
1673
1674         if (attr.fa_src == 0 || attr.fa_dst == 0) {
1675                 fprintf(stderr,
1676                         "Please provide both source and destination of %s rule\n",
1677                         name);
1678                 return -1;
1679         }
1680
1681         if (attr.fa_local_nid == 0)
1682                 attr.fa_local_nid = LNET_NID_ANY;
1683
1684         data.ioc_flags = opc;
1685         data.ioc_inllen1 = sizeof(attr);
1686         data.ioc_inlbuf1 = (char *)&attr;
1687         if (libcfs_ioctl_pack(&data, &ioc_buf, IOC_BUF_SIZE) != 0) {
1688                 fprintf(stderr, "libcfs_ioctl_pack failed\n");
1689                 return -1;
1690         }
1691
1692         rc = l_ioctl(LNET_DEV_ID, IOC_LIBCFS_LNET_FAULT, ioc_buf);
1693         if (rc != 0) {
1694                 fprintf(stderr, "add %s rule %s->%s failed: %s\n",
1695                         name, libcfs_nid2str(attr.fa_src),
1696                         libcfs_nid2str(attr.fa_dst), strerror(errno));
1697                 return -1;
1698         }
1699
1700         printf("Added %s rule %s->%s (1/%d)\n",
1701                name, libcfs_nid2str(attr.fa_src), libcfs_nid2str(attr.fa_dst),
1702                opc == LNET_CTL_DROP_ADD ?
1703                attr.u.drop.da_rate : attr.u.delay.la_rate);
1704         return 0;
1705
1706 getopt_failed:
1707         optind = 1;
1708         return -1;
1709 }
1710
1711 int
1712 jt_ptl_drop_add(int argc, char **argv)
1713 {
1714         return fault_simul_rule_add(LNET_CTL_DROP_ADD, "drop", argc, argv);
1715 }
1716
1717 int
1718 jt_ptl_delay_add(int argc, char **argv)
1719 {
1720         return fault_simul_rule_add(LNET_CTL_DELAY_ADD, "delay", argc, argv);
1721 }
1722
1723 static int
1724 fault_simul_rule_del(__u32 opc, char *name, int argc, char **argv)
1725 {
1726         struct libcfs_ioctl_data data = { { 0 } };
1727         struct lnet_fault_attr   attr;
1728         bool all = false;
1729         int rc;
1730
1731         static const struct option opts[] = {
1732                 { .name = "source", .has_arg = required_argument, .val = 's' },
1733                 { .name = "dest",   .has_arg = required_argument, .val = 'd' },
1734                 { .name = "all",    .has_arg = no_argument,       .val = 'a' },
1735                 { .name = NULL } };
1736
1737         if (argc == 1) {
1738                 fprintf(stderr,
1739                         "Failed, please provide source and destination of rule\n");
1740                 return -1;
1741         }
1742
1743         memset(&attr, 0, sizeof(attr));
1744         while (1) {
1745                 int c = getopt_long(argc, argv, "s:d:a", opts, NULL);
1746
1747                 if (c == -1 || all)
1748                         break;
1749
1750                 switch (c) {
1751                 case 's':
1752                         rc = fault_attr_nid_parse(optarg, &attr.fa_src);
1753                         if (rc != 0)
1754                                 goto getopt_failed;
1755                         break;
1756                 case 'd':
1757                         rc = fault_attr_nid_parse(optarg, &attr.fa_dst);
1758                         if (rc != 0)
1759                                 goto getopt_failed;
1760                         break;
1761                 case 'a':
1762                         attr.fa_src = 0;
1763                         attr.fa_dst = 0;
1764                         all = true;
1765
1766                         break;
1767                 default:
1768                         fprintf(stderr, "error: %s: option '%s' unrecognized\n",
1769                                 argv[0], argv[optind - 1]);
1770                         goto getopt_failed;
1771                 }
1772         }
1773         optind = 1;
1774
1775         data.ioc_flags = opc;
1776         data.ioc_inllen1 = sizeof(attr);
1777         data.ioc_inlbuf1 = (char *)&attr;
1778         if (libcfs_ioctl_pack(&data, &ioc_buf, IOC_BUF_SIZE) != 0) {
1779                 fprintf(stderr, "libcfs_ioctl_pack failed\n");
1780                 return -1;
1781         }
1782
1783         rc = l_ioctl(LNET_DEV_ID, IOC_LIBCFS_LNET_FAULT, ioc_buf);
1784         if (rc != 0) {
1785                 fprintf(stderr, "remove %s rule %s->%s failed: %s\n", name,
1786                         all ? "all" : libcfs_nid2str(attr.fa_src),
1787                         all ? "all" : libcfs_nid2str(attr.fa_dst),
1788                         strerror(errno));
1789                 return -1;
1790         }
1791
1792         libcfs_ioctl_unpack(&data, ioc_buf);
1793         printf("Removed %d %s rules\n", data.ioc_count, name);
1794         return 0;
1795
1796 getopt_failed:
1797         optind = 1;
1798         return -1;
1799 }
1800
1801 int
1802 jt_ptl_drop_del(int argc, char **argv)
1803 {
1804         return fault_simul_rule_del(LNET_CTL_DROP_DEL, "drop", argc, argv);
1805 }
1806
1807 int
1808 jt_ptl_delay_del(int argc, char **argv)
1809 {
1810         return fault_simul_rule_del(LNET_CTL_DELAY_DEL, "delay", argc, argv);
1811 }
1812
1813 static int
1814 fault_simul_rule_reset(__u32 opc, char *name, int argc, char **argv)
1815 {
1816         struct libcfs_ioctl_data   data = { { 0 } };
1817         int                        rc;
1818
1819         LIBCFS_IOC_INIT(data);
1820         data.ioc_flags = opc;
1821
1822         rc = l_ioctl(LNET_DEV_ID, IOC_LIBCFS_LNET_FAULT, &data);
1823         if (rc != 0) {
1824                 fprintf(stderr, "failed to reset %s stats: %s\n",
1825                         name, strerror(errno));
1826                 return -1;
1827         }
1828         return 0;
1829 }
1830
1831 int
1832 jt_ptl_drop_reset(int argc, char **argv)
1833 {
1834         return fault_simul_rule_reset(LNET_CTL_DROP_RESET, "drop", argc, argv);
1835 }
1836
1837 int
1838 jt_ptl_delay_reset(int argc, char **argv)
1839 {
1840         return fault_simul_rule_reset(LNET_CTL_DELAY_RESET, "delay",
1841                                       argc, argv);
1842 }
1843
1844 static int
1845 fault_simul_rule_list(__u32 opc, char *name, int argc, char **argv)
1846 {
1847         struct libcfs_ioctl_data data = { { 0 } };
1848         struct lnet_fault_attr   attr;
1849         struct lnet_fault_stat   stat;
1850         int pos;
1851
1852         printf("LNet %s rules:\n", name);
1853         for (pos = 0;; pos++) {
1854                 int rc;
1855
1856                 memset(&attr, 0, sizeof(attr));
1857                 memset(&stat, 0, sizeof(stat));
1858
1859                 data.ioc_count = pos;
1860                 data.ioc_flags = opc;
1861                 data.ioc_inllen1 = sizeof(attr);
1862                 data.ioc_inlbuf1 = (char *)&attr;
1863                 data.ioc_inllen2 = sizeof(stat);
1864                 data.ioc_inlbuf2 = (char *)&stat;
1865                 if (libcfs_ioctl_pack(&data, &ioc_buf, IOC_BUF_SIZE) != 0) {
1866                         fprintf(stderr, "libcfs_ioctl_pack failed\n");
1867                         return -1;
1868                 }
1869
1870                 rc = l_ioctl(LNET_DEV_ID, IOC_LIBCFS_LNET_FAULT, ioc_buf);
1871                 if (rc != 0)
1872                         break;
1873
1874                 libcfs_ioctl_unpack(&data, ioc_buf);
1875
1876                 if (opc == LNET_CTL_DROP_LIST) {
1877                         printf("%s->%s (1/%d | %d) ptl %#jx, msg %x, %ju/%ju, PUT %ju, ACK %ju, GET %ju, REP %ju\n",
1878                                libcfs_nid2str(attr.fa_src),
1879                                libcfs_nid2str(attr.fa_dst),
1880                                attr.u.drop.da_rate, attr.u.drop.da_interval,
1881                                (uintmax_t)attr.fa_ptl_mask, attr.fa_msg_mask,
1882                                (uintmax_t)stat.u.drop.ds_dropped,
1883                                (uintmax_t)stat.fs_count,
1884                                (uintmax_t)stat.fs_put,
1885                                (uintmax_t)stat.fs_ack,
1886                                (uintmax_t)stat.fs_get,
1887                                (uintmax_t)stat.fs_reply);
1888
1889                 } else if (opc == LNET_CTL_DELAY_LIST) {
1890                         printf("%s->%s (1/%d | %d, latency %d) ptl %#jx, msg %x, %ju/%ju, PUT %ju, ACK %ju, GET %ju, REP %ju\n",
1891                                libcfs_nid2str(attr.fa_src),
1892                                libcfs_nid2str(attr.fa_dst),
1893                                attr.u.delay.la_rate, attr.u.delay.la_interval,
1894                                attr.u.delay.la_latency,
1895                                (uintmax_t)attr.fa_ptl_mask, attr.fa_msg_mask,
1896                                (uintmax_t)stat.u.delay.ls_delayed,
1897                                (uintmax_t)stat.fs_count,
1898                                (uintmax_t)stat.fs_put,
1899                                (uintmax_t)stat.fs_ack,
1900                                (uintmax_t)stat.fs_get,
1901                                (uintmax_t)stat.fs_reply);
1902                 }
1903         }
1904         printf("found total %d\n", pos);
1905
1906         return 0;
1907 }
1908
1909 int
1910 jt_ptl_drop_list(int argc, char **argv)
1911 {
1912         return fault_simul_rule_list(LNET_CTL_DROP_LIST, "drop", argc, argv);
1913 }
1914
1915 int
1916 jt_ptl_delay_list(int argc, char **argv)
1917 {
1918         return fault_simul_rule_list(LNET_CTL_DELAY_LIST, "delay", argc, argv);
1919 }
1920
1921 double
1922 get_cycles_per_usec()
1923 {
1924         FILE *f = fopen("/proc/cpuinfo", "r");
1925         double mhz;
1926         char line[64];
1927
1928         if (f) {
1929                 while (fgets(line, sizeof(line), f))
1930                         if (sscanf(line, "cpu MHz : %lf", &mhz) == 1) {
1931                                 fclose(f);
1932                                 return mhz;
1933                         }
1934                 fclose(f);
1935         }
1936
1937         fprintf(stderr, "Can't read/parse /proc/cpuinfo\n");
1938         return 1000.0;
1939 }
1940
1941 int jt_ptl_testprotocompat(int argc, char **argv)
1942 {
1943         struct libcfs_ioctl_data  data;
1944         int rc;
1945         int flags;
1946         char *end;
1947
1948         if (argc < 2)  {
1949                 fprintf(stderr, "usage: %s <number>\n", argv[0]);
1950                 return 0;
1951         }
1952
1953         flags = strtol(argv[1], &end, 0);
1954         if (flags < 0 || *end != 0) {
1955                 fprintf(stderr, "Can't parse flags '%s'\n", argv[1]);
1956                 return -1;
1957         }
1958
1959         LIBCFS_IOC_INIT(data);
1960         data.ioc_flags = flags;
1961         rc = l_ioctl(LNET_DEV_ID, IOC_LIBCFS_TESTPROTOCOMPAT, &data);
1962
1963         if (rc != 0) {
1964                 fprintf(stderr, "test proto compat %x failed: %s\n",
1965                         flags, strerror(errno));
1966                 return -1;
1967         }
1968
1969         printf("test proto compat %x OK\n", flags);
1970         return 0;
1971 }