/* * lib/route/cls_api.c Classifier Object * * 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 */ /** * @ingroup cls * @defgroup cls_obj Classifier Object * @{ */ #include #include #include #include #include #include #include #include /** @cond SKIP */ #define CLS_ATTR_PRIO (TCA_ATTR_MAX << 1) #define CLS_ATTR_PROTOCOL (TCA_ATTR_MAX << 2) /** @endcond */ static void cls_free_data(struct nl_object *obj) { struct rtnl_cls *cls = (struct rtnl_cls *) obj; struct rtnl_cls_ops *cops; tca_free_data((struct rtnl_tca *) cls); cops = rtnl_cls_lookup_ops(cls); if (cops && cops->co_free_data) cops->co_free_data(cls); nl_data_free(cls->c_subdata); } static int cls_clone(struct nl_object *_dst, struct nl_object *_src) { struct rtnl_cls *dst = nl_object_priv(_dst); struct rtnl_cls *src = nl_object_priv(_src); struct rtnl_cls_ops *cops; int err; err = tca_clone((struct rtnl_tca *) dst, (struct rtnl_tca *) src); if (err < 0) goto errout; if (src->c_subdata) { if (!(dst->c_subdata = nl_data_clone(src->c_subdata))) { err = -NLE_NOMEM; goto errout; } } cops = rtnl_cls_lookup_ops(src); if (cops && cops->co_clone) err = cops->co_clone(dst, src); errout: return err; } static void cls_dump_line(struct nl_object *obj, struct nl_dump_params *p) { char buf[32]; struct rtnl_cls *cls = (struct rtnl_cls *) obj; struct rtnl_cls_ops *cops; tca_dump_line((struct rtnl_tca *) cls, "cls", p); nl_dump(p, " prio %u protocol %s", cls->c_prio, nl_ether_proto2str(cls->c_protocol, buf, sizeof(buf))); cops = rtnl_cls_lookup_ops(cls); if (cops && cops->co_dump[NL_DUMP_LINE]) cops->co_dump[NL_DUMP_LINE](cls, p); nl_dump(p, "\n"); } static void cls_dump_details(struct nl_object *obj, struct nl_dump_params *p) { struct rtnl_cls *cls = (struct rtnl_cls *) obj; struct rtnl_cls_ops *cops; cls_dump_line(obj, p); tca_dump_details((struct rtnl_tca *) cls, p); cops = rtnl_cls_lookup_ops(cls); if (cops && cops->co_dump[NL_DUMP_DETAILS]) cops->co_dump[NL_DUMP_DETAILS](cls, p); else nl_dump(p, "no options\n"); } static void cls_dump_stats(struct nl_object *obj, struct nl_dump_params *p) { struct rtnl_cls *cls = (struct rtnl_cls *) obj; struct rtnl_cls_ops *cops; cls_dump_details(obj, p); tca_dump_stats((struct rtnl_tca *) cls, p); nl_dump(p, "\n"); cops = rtnl_cls_lookup_ops(cls); if (cops && cops->co_dump[NL_DUMP_STATS]) cops->co_dump[NL_DUMP_STATS](cls, p); } /** * @name Allocation/Freeing * @{ */ struct rtnl_cls *rtnl_cls_alloc(void) { return (struct rtnl_cls *) nl_object_alloc(&cls_obj_ops); } void rtnl_cls_put(struct rtnl_cls *cls) { nl_object_put((struct nl_object *) cls); } /** @} */ /** * @name Attributes * @{ */ void rtnl_cls_set_ifindex(struct rtnl_cls *f, int ifindex) { tca_set_ifindex((struct rtnl_tca *) f, ifindex); } int rtnl_cls_get_ifindex(struct rtnl_cls *cls) { return cls->c_ifindex; } void rtnl_cls_set_handle(struct rtnl_cls *f, uint32_t handle) { tca_set_handle((struct rtnl_tca *) f, handle); } void rtnl_cls_set_parent(struct rtnl_cls *f, uint32_t parent) { tca_set_parent((struct rtnl_tca *) f, parent); } uint32_t rtnl_cls_get_parent(struct rtnl_cls *cls) { return cls->c_parent; } int rtnl_cls_set_kind(struct rtnl_cls *cls, const char *kind) { if (cls->ce_mask & TCA_ATTR_KIND) return -NLE_EXIST; tca_set_kind((struct rtnl_tca *) cls, kind); /* Force allocation of data */ rtnl_cls_data(cls); return 0; } struct rtnl_cls_ops *rtnl_cls_get_ops(struct rtnl_cls *cls) { return cls->c_ops; } void rtnl_cls_set_prio(struct rtnl_cls *cls, uint16_t prio) { cls->c_prio = prio; cls->ce_mask |= CLS_ATTR_PRIO; } uint16_t rtnl_cls_get_prio(struct rtnl_cls *cls) { if (cls->ce_mask & CLS_ATTR_PRIO) return cls->c_prio; else return 0; } void rtnl_cls_set_protocol(struct rtnl_cls *cls, uint16_t protocol) { cls->c_protocol = protocol; cls->ce_mask |= CLS_ATTR_PROTOCOL; } uint16_t rtnl_cls_get_protocol(struct rtnl_cls *cls) { if (cls->ce_mask & CLS_ATTR_PROTOCOL) return cls->c_protocol; else return ETH_P_ALL; } void *rtnl_cls_data(struct rtnl_cls *cls) { if (!cls->c_subdata) { struct rtnl_cls_ops *ops = cls->c_ops; if (!ops) { if (!cls->c_kind[0]) BUG(); ops = __rtnl_cls_lookup_ops(cls->c_kind); if (ops == NULL) return NULL; cls->c_ops = ops; } if (!ops->co_size) BUG(); if (!(cls->c_subdata = nl_data_alloc(NULL, ops->co_size))) return NULL; } return nl_data_get(cls->c_subdata); } /** @} */ struct nl_object_ops cls_obj_ops = { .oo_name = "route/cls", .oo_size = sizeof(struct rtnl_cls), .oo_free_data = cls_free_data, .oo_clone = cls_clone, .oo_dump = { [NL_DUMP_LINE] = cls_dump_line, [NL_DUMP_DETAILS] = cls_dump_details, [NL_DUMP_STATS] = cls_dump_stats, }, .oo_compare = tca_compare, .oo_id_attrs = (TCA_ATTR_IFINDEX | TCA_ATTR_HANDLE), }; /** @} */