libnftnl  1.0.8
payload.c
1 /*
2  * (C) 2012 by Pablo Neira Ayuso <pablo@netfilter.org>
3  *
4  * This program is free software; you can redistribute it and/or modify
5  * it under the terms of the GNU General Public License as published
6  * by the Free Software Foundation; either version 2 of the License, or
7  * (at your option) any later version.
8  *
9  * This code has been sponsored by Sophos Astaro <http://www.sophos.com>
10  */
11 
12 #include "internal.h"
13 
14 #include <stdio.h>
15 #include <stdint.h>
16 #include <string.h>
17 #include <limits.h>
18 #include <arpa/inet.h>
19 #include <errno.h>
20 #include <libmnl/libmnl.h>
21 
22 #include <linux/netfilter/nf_tables.h>
23 
24 #include <libnftnl/expr.h>
25 #include <libnftnl/rule.h>
26 
28  enum nft_registers sreg;
29  enum nft_registers dreg;
30  enum nft_payload_bases base;
31  uint32_t offset;
32  uint32_t len;
33  uint32_t csum_type;
34  uint32_t csum_offset;
35  uint32_t csum_flags;
36 };
37 
38 static int
39 nftnl_expr_payload_set(struct nftnl_expr *e, uint16_t type,
40  const void *data, uint32_t data_len)
41 {
42  struct nftnl_expr_payload *payload = nftnl_expr_data(e);
43 
44  switch(type) {
45  case NFTNL_EXPR_PAYLOAD_SREG:
46  payload->sreg = *((uint32_t *)data);
47  break;
48  case NFTNL_EXPR_PAYLOAD_DREG:
49  payload->dreg = *((uint32_t *)data);
50  break;
51  case NFTNL_EXPR_PAYLOAD_BASE:
52  payload->base = *((uint32_t *)data);
53  break;
54  case NFTNL_EXPR_PAYLOAD_OFFSET:
55  payload->offset = *((unsigned int *)data);
56  break;
57  case NFTNL_EXPR_PAYLOAD_LEN:
58  payload->len = *((unsigned int *)data);
59  break;
60  case NFTNL_EXPR_PAYLOAD_CSUM_TYPE:
61  payload->csum_type = *((uint32_t *)data);
62  break;
63  case NFTNL_EXPR_PAYLOAD_CSUM_OFFSET:
64  payload->csum_offset = *((uint32_t *)data);
65  break;
66  case NFTNL_EXPR_PAYLOAD_FLAGS:
67  payload->csum_flags = *((uint32_t *)data);
68  break;
69  default:
70  return -1;
71  }
72  return 0;
73 }
74 
75 static const void *
76 nftnl_expr_payload_get(const struct nftnl_expr *e, uint16_t type,
77  uint32_t *data_len)
78 {
79  struct nftnl_expr_payload *payload = nftnl_expr_data(e);
80 
81  switch(type) {
82  case NFTNL_EXPR_PAYLOAD_SREG:
83  *data_len = sizeof(payload->sreg);
84  return &payload->sreg;
85  case NFTNL_EXPR_PAYLOAD_DREG:
86  *data_len = sizeof(payload->dreg);
87  return &payload->dreg;
88  case NFTNL_EXPR_PAYLOAD_BASE:
89  *data_len = sizeof(payload->base);
90  return &payload->base;
91  case NFTNL_EXPR_PAYLOAD_OFFSET:
92  *data_len = sizeof(payload->offset);
93  return &payload->offset;
94  case NFTNL_EXPR_PAYLOAD_LEN:
95  *data_len = sizeof(payload->len);
96  return &payload->len;
97  case NFTNL_EXPR_PAYLOAD_CSUM_TYPE:
98  *data_len = sizeof(payload->csum_type);
99  return &payload->csum_type;
100  case NFTNL_EXPR_PAYLOAD_CSUM_OFFSET:
101  *data_len = sizeof(payload->csum_offset);
102  return &payload->csum_offset;
103  case NFTNL_EXPR_PAYLOAD_FLAGS:
104  *data_len = sizeof(payload->csum_flags);
105  return &payload->csum_flags;
106  }
107  return NULL;
108 }
109 
110 static int nftnl_expr_payload_cb(const struct nlattr *attr, void *data)
111 {
112  const struct nlattr **tb = data;
113  int type = mnl_attr_get_type(attr);
114 
115  if (mnl_attr_type_valid(attr, NFTA_PAYLOAD_MAX) < 0)
116  return MNL_CB_OK;
117 
118  switch(type) {
119  case NFTA_PAYLOAD_SREG:
120  case NFTA_PAYLOAD_DREG:
121  case NFTA_PAYLOAD_BASE:
122  case NFTA_PAYLOAD_OFFSET:
123  case NFTA_PAYLOAD_LEN:
124  case NFTA_PAYLOAD_CSUM_TYPE:
125  case NFTA_PAYLOAD_CSUM_OFFSET:
126  case NFTA_PAYLOAD_CSUM_FLAGS:
127  if (mnl_attr_validate(attr, MNL_TYPE_U32) < 0)
128  abi_breakage();
129  break;
130  }
131 
132  tb[type] = attr;
133  return MNL_CB_OK;
134 }
135 
136 static void
137 nftnl_expr_payload_build(struct nlmsghdr *nlh, const struct nftnl_expr *e)
138 {
139  struct nftnl_expr_payload *payload = nftnl_expr_data(e);
140 
141  if (e->flags & (1 << NFTNL_EXPR_PAYLOAD_SREG))
142  mnl_attr_put_u32(nlh, NFTA_PAYLOAD_SREG, htonl(payload->sreg));
143  if (e->flags & (1 << NFTNL_EXPR_PAYLOAD_DREG))
144  mnl_attr_put_u32(nlh, NFTA_PAYLOAD_DREG, htonl(payload->dreg));
145  if (e->flags & (1 << NFTNL_EXPR_PAYLOAD_BASE))
146  mnl_attr_put_u32(nlh, NFTA_PAYLOAD_BASE, htonl(payload->base));
147  if (e->flags & (1 << NFTNL_EXPR_PAYLOAD_OFFSET))
148  mnl_attr_put_u32(nlh, NFTA_PAYLOAD_OFFSET, htonl(payload->offset));
149  if (e->flags & (1 << NFTNL_EXPR_PAYLOAD_LEN))
150  mnl_attr_put_u32(nlh, NFTA_PAYLOAD_LEN, htonl(payload->len));
151  if (e->flags & (1 << NFTNL_EXPR_PAYLOAD_CSUM_TYPE))
152  mnl_attr_put_u32(nlh, NFTA_PAYLOAD_CSUM_TYPE,
153  htonl(payload->csum_type));
154  if (e->flags & (1 << NFTNL_EXPR_PAYLOAD_CSUM_OFFSET))
155  mnl_attr_put_u32(nlh, NFTA_PAYLOAD_CSUM_OFFSET,
156  htonl(payload->csum_offset));
157  if (e->flags & (1 << NFTNL_EXPR_PAYLOAD_FLAGS))
158  mnl_attr_put_u32(nlh, NFTA_PAYLOAD_CSUM_FLAGS,
159  htonl(payload->csum_flags));
160 }
161 
162 static int
163 nftnl_expr_payload_parse(struct nftnl_expr *e, struct nlattr *attr)
164 {
165  struct nftnl_expr_payload *payload = nftnl_expr_data(e);
166  struct nlattr *tb[NFTA_PAYLOAD_MAX+1] = {};
167 
168  if (mnl_attr_parse_nested(attr, nftnl_expr_payload_cb, tb) < 0)
169  return -1;
170 
171  if (tb[NFTA_PAYLOAD_SREG]) {
172  payload->sreg = ntohl(mnl_attr_get_u32(tb[NFTA_PAYLOAD_SREG]));
173  e->flags |= (1 << NFTNL_EXPR_PAYLOAD_SREG);
174  }
175  if (tb[NFTA_PAYLOAD_DREG]) {
176  payload->dreg = ntohl(mnl_attr_get_u32(tb[NFTA_PAYLOAD_DREG]));
177  e->flags |= (1 << NFTNL_EXPR_PAYLOAD_DREG);
178  }
179  if (tb[NFTA_PAYLOAD_BASE]) {
180  payload->base = ntohl(mnl_attr_get_u32(tb[NFTA_PAYLOAD_BASE]));
181  e->flags |= (1 << NFTNL_EXPR_PAYLOAD_BASE);
182  }
183  if (tb[NFTA_PAYLOAD_OFFSET]) {
184  payload->offset = ntohl(mnl_attr_get_u32(tb[NFTA_PAYLOAD_OFFSET]));
185  e->flags |= (1 << NFTNL_EXPR_PAYLOAD_OFFSET);
186  }
187  if (tb[NFTA_PAYLOAD_LEN]) {
188  payload->len = ntohl(mnl_attr_get_u32(tb[NFTA_PAYLOAD_LEN]));
189  e->flags |= (1 << NFTNL_EXPR_PAYLOAD_LEN);
190  }
191  if (tb[NFTA_PAYLOAD_CSUM_TYPE]) {
192  payload->csum_type = ntohl(mnl_attr_get_u32(tb[NFTA_PAYLOAD_CSUM_TYPE]));
193  e->flags |= (1 << NFTNL_EXPR_PAYLOAD_CSUM_TYPE);
194  }
195  if (tb[NFTA_PAYLOAD_CSUM_OFFSET]) {
196  payload->csum_offset = ntohl(mnl_attr_get_u32(tb[NFTA_PAYLOAD_CSUM_OFFSET]));
197  e->flags |= (1 << NFTNL_EXPR_PAYLOAD_CSUM_OFFSET);
198  }
199  if (tb[NFTA_PAYLOAD_CSUM_FLAGS]) {
200  payload->csum_flags = ntohl(mnl_attr_get_u32(tb[NFTA_PAYLOAD_CSUM_FLAGS]));
201  e->flags |= (1 << NFTNL_EXPR_PAYLOAD_FLAGS);
202  }
203  return 0;
204 }
205 
206 static const char *base2str_array[NFT_PAYLOAD_TRANSPORT_HEADER+1] = {
207  [NFT_PAYLOAD_LL_HEADER] = "link",
208  [NFT_PAYLOAD_NETWORK_HEADER] = "network",
209  [NFT_PAYLOAD_TRANSPORT_HEADER] = "transport",
210 };
211 
212 static const char *base2str(enum nft_payload_bases base)
213 {
214  if (base > NFT_PAYLOAD_TRANSPORT_HEADER)
215  return "unknown";
216 
217  return base2str_array[base];
218 }
219 
220 static inline int nftnl_str2base(const char *base)
221 {
222  if (strcmp(base, "link") == 0)
223  return NFT_PAYLOAD_LL_HEADER;
224  else if (strcmp(base, "network") == 0)
225  return NFT_PAYLOAD_NETWORK_HEADER;
226  else if (strcmp(base, "transport") == 0)
227  return NFT_PAYLOAD_TRANSPORT_HEADER;
228  else {
229  errno = EINVAL;
230  return -1;
231  }
232 }
233 
234 static int
235 nftnl_expr_payload_json_parse(struct nftnl_expr *e, json_t *root,
236  struct nftnl_parse_err *err)
237 {
238 #ifdef JSON_PARSING
239  const char *base_str;
240  uint32_t reg, uval32;
241  int base;
242 
243  if (nftnl_jansson_parse_reg(root, "dreg", NFTNL_TYPE_U32, &reg, err) == 0)
244  nftnl_expr_set_u32(e, NFTNL_EXPR_PAYLOAD_DREG, reg);
245 
246  base_str = nftnl_jansson_parse_str(root, "base", err);
247  if (base_str != NULL) {
248  base = nftnl_str2base(base_str);
249  if (base < 0)
250  return -1;
251 
252  nftnl_expr_set_u32(e, NFTNL_EXPR_PAYLOAD_BASE, base);
253  }
254 
255  if (nftnl_jansson_parse_val(root, "offset", NFTNL_TYPE_U32, &uval32,
256  err) == 0)
257  nftnl_expr_set_u32(e, NFTNL_EXPR_PAYLOAD_OFFSET, uval32);
258 
259  if (nftnl_jansson_parse_val(root, "len", NFTNL_TYPE_U32, &uval32, err) == 0)
260  nftnl_expr_set_u32(e, NFTNL_EXPR_PAYLOAD_LEN, uval32);
261 
262  return 0;
263 #else
264  errno = EOPNOTSUPP;
265  return -1;
266 #endif
267 }
268 
269 static int nftnl_expr_payload_export(char *buf, size_t size, uint32_t flags,
270  const struct nftnl_expr *e, int type)
271 {
272  struct nftnl_expr_payload *payload = nftnl_expr_data(e);
273  NFTNL_BUF_INIT(b, buf, size);
274 
275  if (e->flags & (1 << NFTNL_EXPR_PAYLOAD_DREG))
276  nftnl_buf_u32(&b, type, payload->dreg, DREG);
277  if (e->flags & (1 << NFTNL_EXPR_PAYLOAD_OFFSET))
278  nftnl_buf_u32(&b, type, payload->offset, OFFSET);
279  if (e->flags & (1 << NFTNL_EXPR_PAYLOAD_LEN))
280  nftnl_buf_u32(&b, type, payload->len, LEN);
281  if (e->flags & (1 << NFTNL_EXPR_PAYLOAD_BASE))
282  nftnl_buf_str(&b, type, base2str(payload->base), BASE);
283 
284  return nftnl_buf_done(&b);
285 }
286 
287 static int
288 nftnl_expr_payload_snprintf(char *buf, size_t len, uint32_t type,
289  uint32_t flags, const struct nftnl_expr *e)
290 {
291  struct nftnl_expr_payload *payload = nftnl_expr_data(e);
292 
293  switch (type) {
294  case NFTNL_OUTPUT_DEFAULT:
295  if (payload->sreg)
296  return snprintf(buf, len, "write reg %u => %ub @ %s header + %u csum_type %u csum_off %u csum_flags 0x%x ",
297  payload->sreg,
298  payload->len, base2str(payload->base),
299  payload->offset, payload->csum_type,
300  payload->csum_offset,
301  payload->csum_flags);
302  else
303  return snprintf(buf, len, "load %ub @ %s header + %u => reg %u ",
304  payload->len, base2str(payload->base),
305  payload->offset, payload->dreg);
306  case NFTNL_OUTPUT_XML:
307  case NFTNL_OUTPUT_JSON:
308  return nftnl_expr_payload_export(buf, len, flags, e, type);
309  default:
310  break;
311  }
312  return -1;
313 }
314 
315 static bool nftnl_expr_payload_cmp(const struct nftnl_expr *e1,
316  const struct nftnl_expr *e2)
317 {
318  struct nftnl_expr_payload *p1 = nftnl_expr_data(e1);
319  struct nftnl_expr_payload *p2 = nftnl_expr_data(e2);
320  bool eq = true;
321 
322  if (e1->flags & (1 << NFTNL_EXPR_PAYLOAD_SREG))
323  eq &= (p1->sreg == p2->sreg);
324  if (e1->flags & (1 << NFTNL_EXPR_PAYLOAD_DREG))
325  eq &= (p1->dreg == p2->dreg);
326  if (e1->flags & (1 << NFTNL_EXPR_PAYLOAD_BASE))
327  eq &= (p1->base == p2->base);
328  if (e1->flags & (1 << NFTNL_EXPR_PAYLOAD_OFFSET))
329  eq &= (p1->offset == p2->offset);
330  if (e1->flags & (1 << NFTNL_EXPR_PAYLOAD_LEN))
331  eq &= (p1->len == p2->len);
332  if (e1->flags & (1 << NFTNL_EXPR_PAYLOAD_CSUM_TYPE))
333  eq &= (p1->csum_type == p2->csum_type);
334  if (e1->flags & (1 << NFTNL_EXPR_PAYLOAD_CSUM_OFFSET))
335  eq &= (p1->csum_offset == p2->csum_offset);
336  if (e1->flags & (1 << NFTNL_EXPR_PAYLOAD_FLAGS))
337  eq &= (p1->csum_flags == p2->csum_flags);
338 
339  return eq;
340 }
341 
342 struct expr_ops expr_ops_payload = {
343  .name = "payload",
344  .alloc_len = sizeof(struct nftnl_expr_payload),
345  .max_attr = NFTA_PAYLOAD_MAX,
346  .cmp = nftnl_expr_payload_cmp,
347  .set = nftnl_expr_payload_set,
348  .get = nftnl_expr_payload_get,
349  .parse = nftnl_expr_payload_parse,
350  .build = nftnl_expr_payload_build,
351  .snprintf = nftnl_expr_payload_snprintf,
352  .json_parse = nftnl_expr_payload_json_parse,
353 };