nexmon – Rev 1

Subversion Repositories:
Rev:
/*
 * lib/route/nexthop.c  Routing Nexthop
 *
 *      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-2008 Thomas Graf <tgraf@suug.ch>
 */

/**
 * @ingroup route_obj
 * @defgroup nexthop Nexthop
 * @{
 */

#include <netlink-local.h>
#include <netlink/netlink.h>
#include <netlink/utils.h>
#include <netlink/route/rtnl.h>
#include <netlink/route/route.h>

/** @cond SKIP */
#define NH_ATTR_FLAGS   0x000001
#define NH_ATTR_WEIGHT  0x000002
#define NH_ATTR_IFINDEX 0x000004
#define NH_ATTR_GATEWAY 0x000008
#define NH_ATTR_REALMS  0x000010
/** @endcond */

/**
 * @name Allocation/Freeing
 * @{
 */

struct rtnl_nexthop *rtnl_route_nh_alloc(void)
{
        struct rtnl_nexthop *nh;

        nh = calloc(1, sizeof(*nh));
        if (!nh)
                return NULL;

        nl_init_list_head(&nh->rtnh_list);

        return nh;
}

struct rtnl_nexthop *rtnl_route_nh_clone(struct rtnl_nexthop *src)
{
        struct rtnl_nexthop *nh;

        nh = rtnl_route_nh_alloc();
        if (!nh)
                return NULL;

        nh->rtnh_flags = src->rtnh_flags;
        nh->rtnh_flag_mask = src->rtnh_flag_mask;
        nh->rtnh_weight = src->rtnh_weight;
        nh->rtnh_ifindex = src->rtnh_ifindex;
        nh->ce_mask = src->ce_mask;

        if (src->rtnh_gateway) {
                nh->rtnh_gateway = nl_addr_clone(src->rtnh_gateway);
                if (!nh->rtnh_gateway) {
                        free(nh);
                        return NULL;
                }
        }

        return nh;
}

void rtnl_route_nh_free(struct rtnl_nexthop *nh)
{
        nl_addr_put(nh->rtnh_gateway);
        free(nh);
}

/** @} */

int rtnl_route_nh_compare(struct rtnl_nexthop *a, struct rtnl_nexthop *b,
                          uint32_t attrs, int loose)
{
        int diff = 0;

#define NH_DIFF(ATTR, EXPR) ATTR_DIFF(attrs, NH_ATTR_##ATTR, a, b, EXPR)

        diff |= NH_DIFF(IFINDEX,        a->rtnh_ifindex != b->rtnh_ifindex);
        diff |= NH_DIFF(WEIGHT,         a->rtnh_weight != b->rtnh_weight);
        diff |= NH_DIFF(REALMS,         a->rtnh_realms != b->rtnh_realms);
        diff |= NH_DIFF(GATEWAY,        nl_addr_cmp(a->rtnh_gateway,
                                                    b->rtnh_gateway));

        if (loose)
                diff |= NH_DIFF(FLAGS,
                          (a->rtnh_flags ^ b->rtnh_flags) & b->rtnh_flag_mask);
        else
                diff |= NH_DIFF(FLAGS, a->rtnh_flags != b->rtnh_flags);
        
#undef NH_DIFF

        return diff;
}

static void nh_dump_line(struct rtnl_nexthop *nh, struct nl_dump_params *dp)
{
        struct nl_cache *link_cache;
        char buf[128];

        link_cache = nl_cache_mngt_require("route/link");

        nl_dump(dp, "via");

        if (nh->ce_mask & NH_ATTR_GATEWAY)
                nl_dump(dp, " %s", nl_addr2str(nh->rtnh_gateway,
                                                   buf, sizeof(buf)));

        if(nh->ce_mask & NH_ATTR_IFINDEX) {
                if (link_cache) {
                        nl_dump(dp, " dev %s",
                                rtnl_link_i2name(link_cache,
                                                 nh->rtnh_ifindex,
                                                 buf, sizeof(buf)));
                } else
                        nl_dump(dp, " dev %d", nh->rtnh_ifindex);
        }

        nl_dump(dp, " ");
}

static void nh_dump_details(struct rtnl_nexthop *nh, struct nl_dump_params *dp)
{
        struct nl_cache *link_cache;
        char buf[128];

        link_cache = nl_cache_mngt_require("route/link");

        nl_dump(dp, "nexthop");

        if (nh->ce_mask & NH_ATTR_GATEWAY)
                nl_dump(dp, " via %s", nl_addr2str(nh->rtnh_gateway,
                                                   buf, sizeof(buf)));

        if(nh->ce_mask & NH_ATTR_IFINDEX) {
                if (link_cache) {
                        nl_dump(dp, " dev %s",
                                rtnl_link_i2name(link_cache,
                                                 nh->rtnh_ifindex,
                                                 buf, sizeof(buf)));
                } else
                        nl_dump(dp, " dev %d", nh->rtnh_ifindex);
        }

        if (nh->ce_mask & NH_ATTR_WEIGHT)
                nl_dump(dp, " weight %u", nh->rtnh_weight);

        if (nh->ce_mask & NH_ATTR_REALMS)
                nl_dump(dp, " realm %04x:%04x",
                        RTNL_REALM_FROM(nh->rtnh_realms),
                        RTNL_REALM_TO(nh->rtnh_realms));

        if (nh->ce_mask & NH_ATTR_FLAGS)
                nl_dump(dp, " <%s>", rtnl_route_nh_flags2str(nh->rtnh_flags,
                                                        buf, sizeof(buf)));
}

static void nh_dump_env(struct rtnl_nexthop *nh, struct nl_dump_params *dp)
{
        struct nl_cache *link_cache;
        char buf[128];

        link_cache = nl_cache_mngt_require("route/link");

        if (nh->ce_mask & NH_ATTR_GATEWAY)
                nl_dump_line(dp, "ROUTE_NH%d_VIA=%s\n", dp->dp_ivar,
                        nl_addr2str(nh->rtnh_gateway, buf, sizeof(buf)));

        if(nh->ce_mask & NH_ATTR_IFINDEX) {
                if (link_cache) {
                        nl_dump_line(dp, "ROUTE_NH%d_DEV=%s\n", dp->dp_ivar,
                                        rtnl_link_i2name(link_cache,
                                                 nh->rtnh_ifindex,
                                                 buf, sizeof(buf)));
                } else
                        nl_dump_line(dp, "ROUTE_NH%d_DEV=%d\n", dp->dp_ivar,
                                        nh->rtnh_ifindex);
        }

        if (nh->ce_mask & NH_ATTR_WEIGHT)
                nl_dump_line(dp, "ROUTE_NH%d_WEIGHT=%u\n", dp->dp_ivar,
                                nh->rtnh_weight);

        if (nh->ce_mask & NH_ATTR_REALMS)
                nl_dump_line(dp, "ROUTE_NH%d_REALM=%04x:%04x\n", dp->dp_ivar,
                        RTNL_REALM_FROM(nh->rtnh_realms),
                        RTNL_REALM_TO(nh->rtnh_realms));

        if (nh->ce_mask & NH_ATTR_FLAGS)
                nl_dump_line(dp, "ROUTE_NH%d_FLAGS=<%s>\n", dp->dp_ivar,
                        rtnl_route_nh_flags2str(nh->rtnh_flags,
                                                        buf, sizeof(buf)));
}
void rtnl_route_nh_dump(struct rtnl_nexthop *nh, struct nl_dump_params *dp)
{
        switch (dp->dp_type) {
        case NL_DUMP_LINE:
                nh_dump_line(nh, dp);
                break;

        case NL_DUMP_DETAILS:
        case NL_DUMP_STATS:
                if (dp->dp_ivar == NH_DUMP_FROM_DETAILS)
                        nh_dump_details(nh, dp);
                break;

        case NL_DUMP_ENV:
                nh_dump_env(nh, dp);
                break;
        
        default:
                break;
        }
}

/**
 * @name Attributes
 * @{
 */

void rtnl_route_nh_set_weight(struct rtnl_nexthop *nh, uint8_t weight)
{
        nh->rtnh_weight = weight;
        nh->ce_mask |= NH_ATTR_WEIGHT;
}

uint8_t rtnl_route_nh_get_weight(struct rtnl_nexthop *nh)
{
        return nh->rtnh_weight;
}

void rtnl_route_nh_set_ifindex(struct rtnl_nexthop *nh, int ifindex)
{
        nh->rtnh_ifindex = ifindex;
        nh->ce_mask |= NH_ATTR_IFINDEX;
}

int rtnl_route_nh_get_ifindex(struct rtnl_nexthop *nh)
{
        return nh->rtnh_ifindex;
}       

void rtnl_route_nh_set_gateway(struct rtnl_nexthop *nh, struct nl_addr *addr)
{
        struct nl_addr *old = nh->rtnh_gateway;

        if (addr) {
                nh->rtnh_gateway = nl_addr_get(addr);
                nh->ce_mask |= NH_ATTR_GATEWAY;
        } else {
                nh->ce_mask &= ~NH_ATTR_GATEWAY;
                nh->rtnh_gateway = NULL;
        }

        if (old)
                nl_addr_put(old);
}

struct nl_addr *rtnl_route_nh_get_gateway(struct rtnl_nexthop *nh)
{
        return nh->rtnh_gateway;
}

void rtnl_route_nh_set_flags(struct rtnl_nexthop *nh, unsigned int flags)
{
        nh->rtnh_flag_mask |= flags;
        nh->rtnh_flags |= flags;
        nh->ce_mask |= NH_ATTR_FLAGS;
}

void rtnl_route_nh_unset_flags(struct rtnl_nexthop *nh, unsigned int flags)
{
        nh->rtnh_flag_mask |= flags;
        nh->rtnh_flags &= ~flags;
        nh->ce_mask |= NH_ATTR_FLAGS;
}

unsigned int rtnl_route_nh_get_flags(struct rtnl_nexthop *nh)
{
        return nh->rtnh_flags;
}

void rtnl_route_nh_set_realms(struct rtnl_nexthop *nh, uint32_t realms)
{
        nh->rtnh_realms = realms;
        nh->ce_mask |= NH_ATTR_REALMS;
}

uint32_t rtnl_route_nh_get_realms(struct rtnl_nexthop *nh)
{
        return nh->rtnh_realms;
}

/** @} */

/**
 * @name Nexthop Flags Translations
 * @{
 */

static struct trans_tbl nh_flags[] = {
        __ADD(RTNH_F_DEAD, dead)
        __ADD(RTNH_F_PERVASIVE, pervasive)
        __ADD(RTNH_F_ONLINK, onlink)
};

char *rtnl_route_nh_flags2str(int flags, char *buf, size_t len)
{
        return __flags2str(flags, buf, len, nh_flags, ARRAY_SIZE(nh_flags));
}

int rtnl_route_nh_str2flags(const char *name)
{
        return __str2flags(name, nh_flags, ARRAY_SIZE(nh_flags));
}

/** @} */

/** @} */