blob: 8b45e7bb07c6824dd6513acf610f957a4e7cd6b9 [file] [log] [blame]
/*
* src/nl-tctree-dump.c Dump Traffic Control Tree
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation version 2.1
* of the License.
*
* Copyright (c) 2003-2006 Thomas Graf <tgraf@suug.ch>
*/
#include "utils.h"
#include <linux/pkt_sched.h>
static struct nl_handle *nl_handle;
static struct nl_cache *qdisc_cache, *class_cache;
static struct nl_dump_params dump_params = {
.dp_type = NL_DUMP_FULL,
};
static int ifindex;
static void print_qdisc(struct nl_object *, void *);
static void print_class(struct nl_object *obj, void *arg)
{
struct rtnl_qdisc *leaf;
struct rtnl_class *class = (struct rtnl_class *) obj;
struct nl_cache *cls_cache;
uint32_t parent = rtnl_class_get_handle(class);
dump_params.dp_prefix = (int)(long) arg;
nl_object_dump(obj, &dump_params);
leaf = rtnl_class_leaf_qdisc(class, qdisc_cache);
if (leaf)
print_qdisc((struct nl_object *) leaf, arg + 2);
rtnl_class_foreach_child(class, class_cache, &print_class, arg + 2);
cls_cache = rtnl_cls_alloc_cache(nl_handle, ifindex, parent);
if (!cls_cache)
return;
dump_params.dp_prefix = (int)(long) arg + 2;
nl_cache_dump(cls_cache, &dump_params);
nl_cache_free(cls_cache);
}
static void print_qdisc(struct nl_object *obj, void *arg)
{
struct rtnl_qdisc *qdisc = (struct rtnl_qdisc *) obj;
struct nl_cache *cls_cache;
uint32_t parent = rtnl_qdisc_get_handle(qdisc);
dump_params.dp_prefix = (int)(long) arg;
nl_object_dump(obj, &dump_params);
rtnl_qdisc_foreach_child(qdisc, class_cache, &print_class, arg + 2);
cls_cache = rtnl_cls_alloc_cache(nl_handle, ifindex, parent);
if (!cls_cache)
return;
dump_params.dp_prefix = (int)(long) arg + 2;
nl_cache_dump(cls_cache, &dump_params);
nl_cache_free(cls_cache);
}
static void print_link(struct nl_object *obj, void *arg)
{
struct rtnl_link *link = (struct rtnl_link *) obj;
struct rtnl_qdisc *qdisc;
ifindex = rtnl_link_get_ifindex(link);
dump_params.dp_prefix = 0;
nl_object_dump(obj, &dump_params);
class_cache = rtnl_class_alloc_cache(nl_handle, ifindex);
if (!class_cache)
return;
qdisc = rtnl_qdisc_get_by_parent(qdisc_cache, ifindex, TC_H_ROOT);
if (qdisc) {
print_qdisc((struct nl_object *) qdisc, (void *) 2);
rtnl_qdisc_put(qdisc);
}
qdisc = rtnl_qdisc_get_by_parent(qdisc_cache, ifindex, 0);
if (qdisc) {
print_qdisc((struct nl_object *) qdisc, (void *) 2);
rtnl_qdisc_put(qdisc);
}
qdisc = rtnl_qdisc_get_by_parent(qdisc_cache, ifindex, TC_H_INGRESS);
if (qdisc) {
print_qdisc((struct nl_object *) qdisc, (void *) 2);
rtnl_qdisc_put(qdisc);
}
nl_cache_free(class_cache);
}
int main(int argc, char *argv[])
{
struct nl_cache *link_cache;
if (nltool_init(argc, argv) < 0)
return -1;
dump_params.dp_fd = stdout;
if (argc > 1) {
if (!strcasecmp(argv[1], "brief"))
dump_params.dp_type = NL_DUMP_BRIEF;
else if (!strcasecmp(argv[1], "full"))
dump_params.dp_type = NL_DUMP_FULL;
else if (!strcasecmp(argv[1], "stats"))
dump_params.dp_type = NL_DUMP_STATS;
}
nl_handle = nltool_alloc_handle();
if (!nl_handle)
return 1;
if (nltool_connect(nl_handle, NETLINK_ROUTE) < 0)
return 1;
link_cache = nltool_alloc_link_cache(nl_handle);
if (!link_cache)
return 1;
qdisc_cache = nltool_alloc_qdisc_cache(nl_handle);
if (!qdisc_cache)
return 1;
nl_cache_foreach(link_cache, &print_link, NULL);
nl_cache_free(qdisc_cache);
nl_cache_free(link_cache);
nl_close(nl_handle);
nl_handle_destroy(nl_handle);
return 0;
}