1 /*
2 * lib/route/cls/ematch/cmp.c Simple packet data comparison ematch
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) 2008-2013 Thomas Graf <tgraf@suug.ch>
10 */
11
12 /**
13 * @ingroup ematch
14 * @defgroup em_cmp Simple packet data comparison
15 *
16 * @{
17 */
18
19 #include <netlink-private/netlink.h>
20 #include <netlink-private/tc.h>
21 #include <netlink/netlink.h>
22 #include <netlink/route/cls/ematch.h>
23 #include <netlink/route/cls/ematch/cmp.h>
24 #include <linux/tc_ematch/tc_em_cmp.h>
25
rtnl_ematch_cmp_set(struct rtnl_ematch * e,struct tcf_em_cmp * cfg)26 void rtnl_ematch_cmp_set(struct rtnl_ematch *e, struct tcf_em_cmp *cfg)
27 {
28 memcpy(rtnl_ematch_data(e), cfg, sizeof(*cfg));
29 }
30
rtnl_ematch_cmp_get(struct rtnl_ematch * e)31 struct tcf_em_cmp *rtnl_ematch_cmp_get(struct rtnl_ematch *e)
32 {
33 return rtnl_ematch_data(e);
34 }
35
cmp_parse(struct rtnl_ematch * e,void * data,size_t len)36 static int cmp_parse(struct rtnl_ematch *e, void *data, size_t len)
37 {
38 memcpy(rtnl_ematch_data(e), data, len);
39
40 return 0;
41 }
42
43 static const char *align_txt[] = {
44 [TCF_EM_ALIGN_U8] = "u8",
45 [TCF_EM_ALIGN_U16] = "u16",
46 [TCF_EM_ALIGN_U32] = "u32"
47 };
48
49 static const char *layer_txt[] = {
50 [TCF_LAYER_LINK] = "eth",
51 [TCF_LAYER_NETWORK] = "ip",
52 [TCF_LAYER_TRANSPORT] = "tcp"
53 };
54
55 static const char *operand_txt[] = {
56 [TCF_EM_OPND_EQ] = "=",
57 [TCF_EM_OPND_LT] = "<",
58 [TCF_EM_OPND_GT] = ">",
59 };
60
cmp_dump(struct rtnl_ematch * e,struct nl_dump_params * p)61 static void cmp_dump(struct rtnl_ematch *e, struct nl_dump_params *p)
62 {
63 struct tcf_em_cmp *cmp = rtnl_ematch_data(e);
64
65 if (cmp->flags & TCF_EM_CMP_TRANS)
66 nl_dump(p, "ntoh%c(", (cmp->align == TCF_EM_ALIGN_U32) ? 'l' : 's');
67
68 nl_dump(p, "%s at %s+%u",
69 align_txt[cmp->align], layer_txt[cmp->layer], cmp->off);
70
71 if (cmp->mask)
72 nl_dump(p, " & 0x%x", cmp->mask);
73
74 if (cmp->flags & TCF_EM_CMP_TRANS)
75 nl_dump(p, ")");
76
77 nl_dump(p, " %s %u", operand_txt[cmp->opnd], cmp->val);
78 }
79
80 static struct rtnl_ematch_ops cmp_ops = {
81 .eo_kind = TCF_EM_CMP,
82 .eo_name = "cmp",
83 .eo_minlen = sizeof(struct tcf_em_cmp),
84 .eo_datalen = sizeof(struct tcf_em_cmp),
85 .eo_parse = cmp_parse,
86 .eo_dump = cmp_dump,
87 };
88
cmp_init(void)89 static void __init cmp_init(void)
90 {
91 rtnl_ematch_register(&cmp_ops);
92 }
93
94 /** @} */
95