libnftnl  1.1.7
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  memcpy(&payload->sreg, data, sizeof(payload->sreg));
47  break;
48  case NFTNL_EXPR_PAYLOAD_DREG:
49  memcpy(&payload->dreg, data, sizeof(payload->dreg));
50  break;
51  case NFTNL_EXPR_PAYLOAD_BASE:
52  memcpy(&payload->base, data, sizeof(payload->base));
53  break;
54  case NFTNL_EXPR_PAYLOAD_OFFSET:
55  memcpy(&payload->offset, data, sizeof(payload->offset));
56  break;
57  case NFTNL_EXPR_PAYLOAD_LEN:
58  memcpy(&payload->len, data, sizeof(payload->len));
59  break;
60  case NFTNL_EXPR_PAYLOAD_CSUM_TYPE:
61  memcpy(&payload->csum_type, data, sizeof(payload->csum_type));
62  break;
63  case NFTNL_EXPR_PAYLOAD_CSUM_OFFSET:
64  memcpy(&payload->csum_offset, data, sizeof(payload->csum_offset));
65  break;
66  case NFTNL_EXPR_PAYLOAD_FLAGS:
67  memcpy(&payload->csum_flags, data, sizeof(payload->csum_flags));
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_snprintf(char *buf, size_t len, uint32_t type,
236  uint32_t flags, const struct nftnl_expr *e)
237 {
238  struct nftnl_expr_payload *payload = nftnl_expr_data(e);
239 
240  switch (type) {
241  case NFTNL_OUTPUT_DEFAULT:
242  if (payload->sreg)
243  return snprintf(buf, len, "write reg %u => %ub @ %s header + %u csum_type %u csum_off %u csum_flags 0x%x ",
244  payload->sreg,
245  payload->len, base2str(payload->base),
246  payload->offset, payload->csum_type,
247  payload->csum_offset,
248  payload->csum_flags);
249  else
250  return snprintf(buf, len, "load %ub @ %s header + %u => reg %u ",
251  payload->len, base2str(payload->base),
252  payload->offset, payload->dreg);
253  case NFTNL_OUTPUT_XML:
254  case NFTNL_OUTPUT_JSON:
255  default:
256  break;
257  }
258  return -1;
259 }
260 
261 struct expr_ops expr_ops_payload = {
262  .name = "payload",
263  .alloc_len = sizeof(struct nftnl_expr_payload),
264  .max_attr = NFTA_PAYLOAD_MAX,
265  .set = nftnl_expr_payload_set,
266  .get = nftnl_expr_payload_get,
267  .parse = nftnl_expr_payload_parse,
268  .build = nftnl_expr_payload_build,
269  .snprintf = nftnl_expr_payload_snprintf,
270 };