]> rtime.felk.cvut.cz Git - lisovros/iproute2_canprio.git/commitdiff
This patch adds ability to monitor tc events similar to ipmonitor.
authorJamal Hadi Salim <hadi@cyberus.ca>
Tue, 8 Aug 2006 18:55:15 +0000 (11:55 -0700)
committerStephen Hemminger <shemminger@osdl.org>
Tue, 8 Aug 2006 18:55:15 +0000 (11:55 -0700)
User runs "tc monitor" (without quotes) and watches events of
addition, deletion and updates from qdiscs, classes, filters and
actions as they happen.

Signed-off-by: Stephen Hemminger <shemminger@osdl.org>
tc/Makefile
tc/m_action.c
tc/tc.c
tc/tc_class.c
tc/tc_common.h
tc/tc_filter.c
tc/tc_monitor.c [new file with mode: 0644]
tc/tc_qdisc.c

index 9d618ffcef30d5b658f842aac94e63ac1eaca743..7640c58eee2365715dbd12dbb8d350b7a2b7ec9b 100644 (file)
@@ -1,6 +1,6 @@
 TCOBJ= tc.o tc_qdisc.o tc_class.o tc_filter.o tc_util.o \
-       m_police.o m_estimator.o m_action.o m_ematch.o \
-       emp_ematch.yacc.o emp_ematch.lex.o
+       tc_monitor.o m_police.o m_estimator.o m_action.o \
+       m_ematch.o emp_ematch.yacc.o emp_ematch.lex.o
 
 include ../Config
 
index b4eff91c727f1cf379c1b8c783fe43b307191cae..c67bc8836b845fb6b84e20448832a84569250c2a 100644 (file)
@@ -313,7 +313,7 @@ tc_print_action(FILE * f, const struct rtattr *arg)
        return 0;
 }
 
-static int do_print_action(const struct sockaddr_nl *who,
+int print_action(const struct sockaddr_nl *who,
                           struct nlmsghdr *n,
                           void *arg)
 {
@@ -333,7 +333,7 @@ static int do_print_action(const struct sockaddr_nl *who,
 
        if (NULL == tb[TCA_ACT_TAB]) {
                if (n->nlmsg_type != RTM_GETACTION)
-                       fprintf(stderr, "do_print_action: NULL kind\n");
+                       fprintf(stderr, "print_action: NULL kind\n");
                return -1;
        }     
 
@@ -454,7 +454,7 @@ int tc_action_gd(int cmd, unsigned flags, int *argc_p, char ***argv_p)
                return 1;
        }
 
-       if (ans && do_print_action(NULL, &req.n, (void*)stdout) < 0) {
+       if (ans && print_action(NULL, &req.n, (void*)stdout) < 0) {
                fprintf(stderr, "Dump terminated\n");
                return 1;
        }
@@ -556,7 +556,7 @@ int tc_act_list_or_flush(int argc, char **argv, int event)
                        perror("Cannot send dump request");
                        return 1;
                }
-               ret = rtnl_dump_filter(&rth, do_print_action, stdout, NULL, NULL);
+               ret = rtnl_dump_filter(&rth, print_action, stdout, NULL, NULL);
        }
 
        if (event == RTM_DELACTION) { 
diff --git a/tc/tc.c b/tc/tc.c
index fa36ee076d8c735d49c4a4a1a1c00a5528a6dea7..789f6f628423fab0df8ad56d75b8a13e673140a6 100644 (file)
--- a/tc/tc.c
+++ b/tc/tc.c
@@ -181,7 +181,7 @@ static void usage(void)
 {
        fprintf(stderr, "Usage: tc [ OPTIONS ] OBJECT { COMMAND | help }\n"
                        "       tc [-force] -batch file\n"
-                       "where  OBJECT := { qdisc | class | filter | action }\n"
+                       "where  OBJECT := { qdisc | class | filter | action | monitor }\n"
                        "       OPTIONS := { -s[tatistics] | -d[etails] | -r[aw] | -b[atch] [file] }\n");
 }
 
@@ -199,6 +199,9 @@ static int do_cmd(int argc, char **argv)
        if (matches(*argv, "actions") == 0)
                return do_action(argc-1, argv+1);
 
+       if (matches(*argv, "monitor") == 0)
+               return do_tcmonitor(argc-1, argv+1);
+
        if (matches(*argv, "help") == 0) {
                usage();
                return 0;
index 93d5def5f4b2b4392c00aa97c148f7858b7f7231..8d93b5a8fb2e72bb5aeddfbaf1136a952d155f95 100644 (file)
@@ -147,7 +147,7 @@ int tc_class_modify(int cmd, unsigned flags, int argc, char **argv)
 int filter_ifindex;
 __u32 filter_qdisc;
 
-static int print_class(const struct sockaddr_nl *who, 
+int print_class(const struct sockaddr_nl *who, 
                       struct nlmsghdr *n, void *arg)
 {
        FILE *fp = (FILE*)arg;
index 7e135824a7c74139324325b4cb9e54579402e3b5..5bfc43e46a7d940143df10c4604435616c948176 100644 (file)
@@ -6,6 +6,11 @@ extern int do_qdisc(int argc, char **argv);
 extern int do_class(int argc, char **argv);
 extern int do_filter(int argc, char **argv);
 extern int do_action(int argc, char **argv);
+extern int do_tcmonitor(int argc, char **argv);
+extern int print_action(const struct sockaddr_nl *who, struct nlmsghdr *n, void *arg);
+extern int print_filter(const struct sockaddr_nl *who, struct nlmsghdr *n, void *arg);
+extern int print_qdisc(const struct sockaddr_nl *who, struct nlmsghdr *n, void *arg);
+ extern int print_class(const struct sockaddr_nl *who, struct nlmsghdr *n, void *arg);
 
 struct tc_estimator;
 extern int parse_estimator(int *p_argc, char ***p_argv, struct tc_estimator *est);
index 58af07722e310795ec0c728ad3a414c50903b4a6..f3319d55296dc203a1526936ac9cf298601fbb6e 100644 (file)
@@ -176,7 +176,7 @@ static int filter_ifindex;
 static __u32 filter_prio;
 static __u32 filter_protocol;
 
-static int print_filter(const struct sockaddr_nl *who,
+int print_filter(const struct sockaddr_nl *who,
                        struct nlmsghdr *n, 
                        void *arg)
 {
diff --git a/tc/tc_monitor.c b/tc/tc_monitor.c
new file mode 100644 (file)
index 0000000..1af6cf0
--- /dev/null
@@ -0,0 +1,110 @@
+/*
+ * tc_monitor.c                "tc monitor".
+ *
+ *             This program is free software; you can redistribute it and/or
+ *             modify it under the terms of the GNU General Public License
+ *             as published by the Free Software Foundation; either version
+ *             2 of the License, or (at your option) any later version.
+ *
+ * Authors:    Jamal Hadi Salim
+ *
+ */
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <unistd.h>
+#include <syslog.h>
+#include <fcntl.h>
+#include <sys/socket.h>
+#include <netinet/in.h>
+#include <arpa/inet.h>
+#include <string.h>
+#include <time.h>
+#include "rt_names.h"
+#include "utils.h"
+#include "tc_util.h"
+#include "tc_common.h"
+
+
+static void usage(void) __attribute__((noreturn));
+
+static void usage(void)
+{
+       fprintf(stderr, "Usage: tc monitor\n");
+       exit(-1);
+}
+
+
+int accept_tcmsg(const struct sockaddr_nl *who, struct nlmsghdr *n, void *arg)
+{
+       FILE *fp = (FILE*)arg;
+
+       if (n->nlmsg_type == RTM_NEWTFILTER || n->nlmsg_type == RTM_DELTFILTER) {
+               print_filter(who, n, arg);
+               return 0;
+       }
+       if (n->nlmsg_type == RTM_NEWTCLASS || n->nlmsg_type == RTM_DELTCLASS) {
+               print_class(who, n, arg);
+               return 0;
+       }
+       if (n->nlmsg_type == RTM_NEWQDISC || n->nlmsg_type == RTM_DELQDISC) {
+               print_qdisc(who, n, arg);
+               return 0;
+       }
+       if (n->nlmsg_type == RTM_GETACTION || n->nlmsg_type == RTM_NEWACTION ||
+           n->nlmsg_type == RTM_DELACTION) {
+               print_action(who, n, arg);
+               return 0;
+       }
+       if (n->nlmsg_type != NLMSG_ERROR && n->nlmsg_type != NLMSG_NOOP &&
+           n->nlmsg_type != NLMSG_DONE) {
+               fprintf(fp, "Unknown message: length %08d type %08x flags %08x\n",
+                       n->nlmsg_len, n->nlmsg_type, n->nlmsg_flags);
+       }
+       return 0;
+}
+
+int do_tcmonitor(int argc, char **argv)
+{
+       struct rtnl_handle rth;
+       char *file = NULL;
+       unsigned groups = RTMGRP_TC;
+
+       while (argc > 0) {
+               if (matches(*argv, "file") == 0) {
+                       NEXT_ARG();
+                       file = *argv;
+               } else {
+                       if (matches(*argv, "help") == 0) {
+                               usage();
+                       } else {
+                               fprintf(stderr, "Argument \"%s\" is unknown, try \"tc monitor help\".\n", *argv);
+                               exit(-1);
+                       }
+               }
+               argc--; argv++;
+       }
+
+       if (file) {
+               FILE *fp;
+               fp = fopen(file, "r");
+               if (fp == NULL) {
+                       perror("Cannot fopen");
+                       exit(-1);
+               }
+               return rtnl_from_file(fp, accept_tcmsg, (void*)stdout);
+       }
+
+       if (rtnl_open(&rth, groups) < 0)
+               exit(1);
+
+       ll_init_map(&rth);
+
+       if (rtnl_listen(&rth, accept_tcmsg, (void*)stdout) < 0) {
+               rtnl_close(&rth);
+               exit(2);
+       }
+
+       rtnl_close(&rth);
+       exit(0);
+}
index fb107f462b69a278e411e79556e04e3709ce885f..50b2649769c4e0ca72c5d8615a01edb8ed037640 100644 (file)
@@ -162,7 +162,7 @@ int tc_qdisc_modify(int cmd, unsigned flags, int argc, char **argv)
 
 static int filter_ifindex;
 
-static int print_qdisc(const struct sockaddr_nl *who, 
+int print_qdisc(const struct sockaddr_nl *who, 
                       struct nlmsghdr *n, 
                       void *arg)
 {