cls_obj.c 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203
  1. /*
  2. * lib/route/cls_api.c Classifier Object
  3. *
  4. * This library is free software; you can redistribute it and/or
  5. * modify it under the terms of the GNU Lesser General Public
  6. * License as published by the Free Software Foundation version 2.1
  7. * of the License.
  8. *
  9. * Copyright (c) 2003-2006 Thomas Graf <tgraf@suug.ch>
  10. */
  11. /**
  12. * @ingroup cls
  13. * @defgroup cls_obj Classifier Object
  14. * @{
  15. */
  16. #include <netlink-local.h>
  17. #include <netlink-tc.h>
  18. #include <netlink/netlink.h>
  19. #include <netlink/utils.h>
  20. #include <netlink/route/tc.h>
  21. #include <netlink/route/classifier.h>
  22. #include <netlink/route/classifier-modules.h>
  23. #include <netlink/route/link.h>
  24. /** @cond SKIP */
  25. #define CLS_ATTR_PRIO (TCA_ATTR_MAX << 1)
  26. #define CLS_ATTR_PROTOCOL (TCA_ATTR_MAX << 2)
  27. /** @endcond */
  28. static void cls_free_data(struct nl_object *obj)
  29. {
  30. struct rtnl_cls *cls = (struct rtnl_cls *) obj;
  31. struct rtnl_cls_ops *cops;
  32. tca_free_data((struct rtnl_tca *) cls);
  33. cops = rtnl_cls_lookup_ops(cls);
  34. if (cops && cops->co_free_data)
  35. cops->co_free_data(cls);
  36. }
  37. static int cls_clone(struct nl_object *_dst, struct nl_object *_src)
  38. {
  39. struct rtnl_cls *dst = nl_object_priv(_dst);
  40. struct rtnl_cls *src = nl_object_priv(_src);
  41. struct rtnl_cls_ops *cops;
  42. int err;
  43. err = tca_clone((struct rtnl_tca *) dst, (struct rtnl_tca *) src);
  44. if (err < 0)
  45. goto errout;
  46. cops = rtnl_cls_lookup_ops(src);
  47. if (cops && cops->co_clone)
  48. err = cops->co_clone(dst, src);
  49. errout:
  50. return err;
  51. }
  52. static int cls_dump_brief(struct nl_object *obj, struct nl_dump_params *p)
  53. {
  54. char buf[32];
  55. struct rtnl_cls *cls = (struct rtnl_cls *) obj;
  56. struct rtnl_cls_ops *cops;
  57. int line;
  58. line = tca_dump_brief((struct rtnl_tca *) cls, "cls", p, 0);
  59. dp_dump(p, " prio %u protocol %s", cls->c_prio,
  60. nl_ether_proto2str(cls->c_protocol, buf, sizeof(buf)));
  61. cops = rtnl_cls_lookup_ops(cls);
  62. if (cops && cops->co_dump[NL_DUMP_BRIEF])
  63. line = cops->co_dump[NL_DUMP_BRIEF](cls, p, line);
  64. dp_dump(p, "\n");
  65. return line;
  66. }
  67. static int cls_dump_full(struct nl_object *obj, struct nl_dump_params *p)
  68. {
  69. struct rtnl_cls *cls = (struct rtnl_cls *) obj;
  70. struct rtnl_cls_ops *cops;
  71. int line;
  72. line = cls_dump_brief(obj, p);
  73. line = tca_dump_full((struct rtnl_tca *) cls, p, line);
  74. cops = rtnl_cls_lookup_ops(cls);
  75. if (cops && cops->co_dump[NL_DUMP_FULL])
  76. line = cops->co_dump[NL_DUMP_FULL](cls, p, line);
  77. else
  78. dp_dump(p, "no options\n");
  79. return line;
  80. }
  81. static int cls_dump_stats(struct nl_object *obj, struct nl_dump_params *p)
  82. {
  83. struct rtnl_cls *cls = (struct rtnl_cls *) obj;
  84. struct rtnl_cls_ops *cops;
  85. int line;
  86. line = cls_dump_full(obj, p);
  87. line = tca_dump_stats((struct rtnl_tca *) cls, p, line);
  88. dp_dump(p, "\n");
  89. cops = rtnl_cls_lookup_ops(cls);
  90. if (cops && cops->co_dump[NL_DUMP_STATS])
  91. line = cops->co_dump[NL_DUMP_STATS](cls, p, line);
  92. return line;
  93. }
  94. /**
  95. * @name Allocation/Freeing
  96. * @{
  97. */
  98. struct rtnl_cls *rtnl_cls_alloc(void)
  99. {
  100. return (struct rtnl_cls *) nl_object_alloc(&cls_obj_ops);
  101. }
  102. void rtnl_cls_put(struct rtnl_cls *cls)
  103. {
  104. nl_object_put((struct nl_object *) cls);
  105. }
  106. /** @} */
  107. /**
  108. * @name Attributes
  109. * @{
  110. */
  111. void rtnl_cls_set_ifindex(struct rtnl_cls *f, int ifindex)
  112. {
  113. tca_set_ifindex((struct rtnl_tca *) f, ifindex);
  114. }
  115. void rtnl_cls_set_handle(struct rtnl_cls *f, uint32_t handle)
  116. {
  117. tca_set_handle((struct rtnl_tca *) f, handle);
  118. }
  119. void rtnl_cls_set_parent(struct rtnl_cls *f, uint32_t parent)
  120. {
  121. tca_set_parent((struct rtnl_tca *) f, parent);
  122. }
  123. void rtnl_cls_set_kind(struct rtnl_cls *f, const char *kind)
  124. {
  125. tca_set_kind((struct rtnl_tca *) f, kind);
  126. f->c_ops = __rtnl_cls_lookup_ops(kind);
  127. }
  128. void rtnl_cls_set_prio(struct rtnl_cls *cls, int prio)
  129. {
  130. cls->c_prio = prio;
  131. cls->ce_mask |= CLS_ATTR_PRIO;
  132. }
  133. int rtnl_cls_get_prio(struct rtnl_cls *cls)
  134. {
  135. if (cls->ce_mask & CLS_ATTR_PRIO)
  136. return cls->c_prio;
  137. else
  138. return 0;
  139. }
  140. void rtnl_cls_set_protocol(struct rtnl_cls *cls, int protocol)
  141. {
  142. cls->c_protocol = protocol;
  143. cls->ce_mask |= CLS_ATTR_PROTOCOL;
  144. }
  145. int rtnl_cls_get_protocol(struct rtnl_cls *cls)
  146. {
  147. if (cls->ce_mask & CLS_ATTR_PROTOCOL)
  148. return cls->c_protocol;
  149. else
  150. return ETH_P_ALL;
  151. }
  152. /** @} */
  153. struct nl_object_ops cls_obj_ops = {
  154. .oo_name = "route/cls",
  155. .oo_size = sizeof(struct rtnl_cls),
  156. .oo_free_data = cls_free_data,
  157. .oo_clone = cls_clone,
  158. .oo_dump[NL_DUMP_BRIEF] = cls_dump_brief,
  159. .oo_dump[NL_DUMP_FULL] = cls_dump_full,
  160. .oo_dump[NL_DUMP_STATS] = cls_dump_stats,
  161. .oo_compare = tca_compare,
  162. .oo_id_attrs = (TCA_ATTR_IFINDEX | TCA_ATTR_HANDLE),
  163. };
  164. /** @} */