1 // SPDX-License-Identifier: GPL-2.0+
2 /*
3 * IPv6 IOAM implementation
4 *
5 * Author:
6 * Justin Iurman <justin.iurman@uliege.be>
7 */
8
9 #include <linux/errno.h>
10 #include <linux/types.h>
11 #include <linux/kernel.h>
12 #include <linux/net.h>
13 #include <linux/ioam6.h>
14 #include <linux/ioam6_genl.h>
15 #include <linux/rhashtable.h>
16
17 #include <net/addrconf.h>
18 #include <net/genetlink.h>
19 #include <net/ioam6.h>
20
ioam6_ns_release(struct ioam6_namespace * ns)21 static void ioam6_ns_release(struct ioam6_namespace *ns)
22 {
23 kfree_rcu(ns, rcu);
24 }
25
ioam6_sc_release(struct ioam6_schema * sc)26 static void ioam6_sc_release(struct ioam6_schema *sc)
27 {
28 kfree_rcu(sc, rcu);
29 }
30
ioam6_free_ns(void * ptr,void * arg)31 static void ioam6_free_ns(void *ptr, void *arg)
32 {
33 struct ioam6_namespace *ns = (struct ioam6_namespace *)ptr;
34
35 if (ns)
36 ioam6_ns_release(ns);
37 }
38
ioam6_free_sc(void * ptr,void * arg)39 static void ioam6_free_sc(void *ptr, void *arg)
40 {
41 struct ioam6_schema *sc = (struct ioam6_schema *)ptr;
42
43 if (sc)
44 ioam6_sc_release(sc);
45 }
46
ioam6_ns_cmpfn(struct rhashtable_compare_arg * arg,const void * obj)47 static int ioam6_ns_cmpfn(struct rhashtable_compare_arg *arg, const void *obj)
48 {
49 const struct ioam6_namespace *ns = obj;
50
51 return (ns->id != *(__be16 *)arg->key);
52 }
53
ioam6_sc_cmpfn(struct rhashtable_compare_arg * arg,const void * obj)54 static int ioam6_sc_cmpfn(struct rhashtable_compare_arg *arg, const void *obj)
55 {
56 const struct ioam6_schema *sc = obj;
57
58 return (sc->id != *(u32 *)arg->key);
59 }
60
61 static const struct rhashtable_params rht_ns_params = {
62 .key_len = sizeof(__be16),
63 .key_offset = offsetof(struct ioam6_namespace, id),
64 .head_offset = offsetof(struct ioam6_namespace, head),
65 .automatic_shrinking = true,
66 .obj_cmpfn = ioam6_ns_cmpfn,
67 };
68
69 static const struct rhashtable_params rht_sc_params = {
70 .key_len = sizeof(u32),
71 .key_offset = offsetof(struct ioam6_schema, id),
72 .head_offset = offsetof(struct ioam6_schema, head),
73 .automatic_shrinking = true,
74 .obj_cmpfn = ioam6_sc_cmpfn,
75 };
76
77 static struct genl_family ioam6_genl_family;
78
79 static const struct nla_policy ioam6_genl_policy_addns[] = {
80 [IOAM6_ATTR_NS_ID] = { .type = NLA_U16 },
81 [IOAM6_ATTR_NS_DATA] = { .type = NLA_U32 },
82 [IOAM6_ATTR_NS_DATA_WIDE] = { .type = NLA_U64 },
83 };
84
85 static const struct nla_policy ioam6_genl_policy_delns[] = {
86 [IOAM6_ATTR_NS_ID] = { .type = NLA_U16 },
87 };
88
89 static const struct nla_policy ioam6_genl_policy_addsc[] = {
90 [IOAM6_ATTR_SC_ID] = { .type = NLA_U32 },
91 [IOAM6_ATTR_SC_DATA] = { .type = NLA_BINARY,
92 .len = IOAM6_MAX_SCHEMA_DATA_LEN },
93 };
94
95 static const struct nla_policy ioam6_genl_policy_delsc[] = {
96 [IOAM6_ATTR_SC_ID] = { .type = NLA_U32 },
97 };
98
99 static const struct nla_policy ioam6_genl_policy_ns_sc[] = {
100 [IOAM6_ATTR_NS_ID] = { .type = NLA_U16 },
101 [IOAM6_ATTR_SC_ID] = { .type = NLA_U32 },
102 [IOAM6_ATTR_SC_NONE] = { .type = NLA_FLAG },
103 };
104
ioam6_genl_addns(struct sk_buff * skb,struct genl_info * info)105 static int ioam6_genl_addns(struct sk_buff *skb, struct genl_info *info)
106 {
107 struct ioam6_pernet_data *nsdata;
108 struct ioam6_namespace *ns;
109 u64 data64;
110 u32 data32;
111 __be16 id;
112 int err;
113
114 if (!info->attrs[IOAM6_ATTR_NS_ID])
115 return -EINVAL;
116
117 id = cpu_to_be16(nla_get_u16(info->attrs[IOAM6_ATTR_NS_ID]));
118 nsdata = ioam6_pernet(genl_info_net(info));
119
120 mutex_lock(&nsdata->lock);
121
122 ns = rhashtable_lookup_fast(&nsdata->namespaces, &id, rht_ns_params);
123 if (ns) {
124 err = -EEXIST;
125 goto out_unlock;
126 }
127
128 ns = kzalloc(sizeof(*ns), GFP_KERNEL);
129 if (!ns) {
130 err = -ENOMEM;
131 goto out_unlock;
132 }
133
134 ns->id = id;
135
136 if (!info->attrs[IOAM6_ATTR_NS_DATA])
137 data32 = IOAM6_U32_UNAVAILABLE;
138 else
139 data32 = nla_get_u32(info->attrs[IOAM6_ATTR_NS_DATA]);
140
141 if (!info->attrs[IOAM6_ATTR_NS_DATA_WIDE])
142 data64 = IOAM6_U64_UNAVAILABLE;
143 else
144 data64 = nla_get_u64(info->attrs[IOAM6_ATTR_NS_DATA_WIDE]);
145
146 ns->data = cpu_to_be32(data32);
147 ns->data_wide = cpu_to_be64(data64);
148
149 err = rhashtable_lookup_insert_fast(&nsdata->namespaces, &ns->head,
150 rht_ns_params);
151 if (err)
152 kfree(ns);
153
154 out_unlock:
155 mutex_unlock(&nsdata->lock);
156 return err;
157 }
158
ioam6_genl_delns(struct sk_buff * skb,struct genl_info * info)159 static int ioam6_genl_delns(struct sk_buff *skb, struct genl_info *info)
160 {
161 struct ioam6_pernet_data *nsdata;
162 struct ioam6_namespace *ns;
163 struct ioam6_schema *sc;
164 __be16 id;
165 int err;
166
167 if (!info->attrs[IOAM6_ATTR_NS_ID])
168 return -EINVAL;
169
170 id = cpu_to_be16(nla_get_u16(info->attrs[IOAM6_ATTR_NS_ID]));
171 nsdata = ioam6_pernet(genl_info_net(info));
172
173 mutex_lock(&nsdata->lock);
174
175 ns = rhashtable_lookup_fast(&nsdata->namespaces, &id, rht_ns_params);
176 if (!ns) {
177 err = -ENOENT;
178 goto out_unlock;
179 }
180
181 sc = rcu_dereference_protected(ns->schema,
182 lockdep_is_held(&nsdata->lock));
183
184 err = rhashtable_remove_fast(&nsdata->namespaces, &ns->head,
185 rht_ns_params);
186 if (err)
187 goto out_unlock;
188
189 if (sc)
190 rcu_assign_pointer(sc->ns, NULL);
191
192 ioam6_ns_release(ns);
193
194 out_unlock:
195 mutex_unlock(&nsdata->lock);
196 return err;
197 }
198
__ioam6_genl_dumpns_element(struct ioam6_namespace * ns,u32 portid,u32 seq,u32 flags,struct sk_buff * skb,u8 cmd)199 static int __ioam6_genl_dumpns_element(struct ioam6_namespace *ns,
200 u32 portid,
201 u32 seq,
202 u32 flags,
203 struct sk_buff *skb,
204 u8 cmd)
205 {
206 struct ioam6_schema *sc;
207 u64 data64;
208 u32 data32;
209 void *hdr;
210
211 hdr = genlmsg_put(skb, portid, seq, &ioam6_genl_family, flags, cmd);
212 if (!hdr)
213 return -ENOMEM;
214
215 data32 = be32_to_cpu(ns->data);
216 data64 = be64_to_cpu(ns->data_wide);
217
218 if (nla_put_u16(skb, IOAM6_ATTR_NS_ID, be16_to_cpu(ns->id)) ||
219 (data32 != IOAM6_U32_UNAVAILABLE &&
220 nla_put_u32(skb, IOAM6_ATTR_NS_DATA, data32)) ||
221 (data64 != IOAM6_U64_UNAVAILABLE &&
222 nla_put_u64_64bit(skb, IOAM6_ATTR_NS_DATA_WIDE,
223 data64, IOAM6_ATTR_PAD)))
224 goto nla_put_failure;
225
226 rcu_read_lock();
227
228 sc = rcu_dereference(ns->schema);
229 if (sc && nla_put_u32(skb, IOAM6_ATTR_SC_ID, sc->id)) {
230 rcu_read_unlock();
231 goto nla_put_failure;
232 }
233
234 rcu_read_unlock();
235
236 genlmsg_end(skb, hdr);
237 return 0;
238
239 nla_put_failure:
240 genlmsg_cancel(skb, hdr);
241 return -EMSGSIZE;
242 }
243
ioam6_genl_dumpns_start(struct netlink_callback * cb)244 static int ioam6_genl_dumpns_start(struct netlink_callback *cb)
245 {
246 struct ioam6_pernet_data *nsdata = ioam6_pernet(sock_net(cb->skb->sk));
247 struct rhashtable_iter *iter = (struct rhashtable_iter *)cb->args[0];
248
249 if (!iter) {
250 iter = kmalloc(sizeof(*iter), GFP_KERNEL);
251 if (!iter)
252 return -ENOMEM;
253
254 cb->args[0] = (long)iter;
255 }
256
257 rhashtable_walk_enter(&nsdata->namespaces, iter);
258
259 return 0;
260 }
261
ioam6_genl_dumpns_done(struct netlink_callback * cb)262 static int ioam6_genl_dumpns_done(struct netlink_callback *cb)
263 {
264 struct rhashtable_iter *iter = (struct rhashtable_iter *)cb->args[0];
265
266 rhashtable_walk_exit(iter);
267 kfree(iter);
268
269 return 0;
270 }
271
ioam6_genl_dumpns(struct sk_buff * skb,struct netlink_callback * cb)272 static int ioam6_genl_dumpns(struct sk_buff *skb, struct netlink_callback *cb)
273 {
274 struct rhashtable_iter *iter;
275 struct ioam6_namespace *ns;
276 int err;
277
278 iter = (struct rhashtable_iter *)cb->args[0];
279 rhashtable_walk_start(iter);
280
281 for (;;) {
282 ns = rhashtable_walk_next(iter);
283
284 if (IS_ERR(ns)) {
285 if (PTR_ERR(ns) == -EAGAIN)
286 continue;
287 err = PTR_ERR(ns);
288 goto done;
289 } else if (!ns) {
290 break;
291 }
292
293 err = __ioam6_genl_dumpns_element(ns,
294 NETLINK_CB(cb->skb).portid,
295 cb->nlh->nlmsg_seq,
296 NLM_F_MULTI,
297 skb,
298 IOAM6_CMD_DUMP_NAMESPACES);
299 if (err)
300 goto done;
301 }
302
303 err = skb->len;
304
305 done:
306 rhashtable_walk_stop(iter);
307 return err;
308 }
309
ioam6_genl_addsc(struct sk_buff * skb,struct genl_info * info)310 static int ioam6_genl_addsc(struct sk_buff *skb, struct genl_info *info)
311 {
312 struct ioam6_pernet_data *nsdata;
313 int len, len_aligned, err;
314 struct ioam6_schema *sc;
315 u32 id;
316
317 if (!info->attrs[IOAM6_ATTR_SC_ID] || !info->attrs[IOAM6_ATTR_SC_DATA])
318 return -EINVAL;
319
320 id = nla_get_u32(info->attrs[IOAM6_ATTR_SC_ID]);
321 nsdata = ioam6_pernet(genl_info_net(info));
322
323 mutex_lock(&nsdata->lock);
324
325 sc = rhashtable_lookup_fast(&nsdata->schemas, &id, rht_sc_params);
326 if (sc) {
327 err = -EEXIST;
328 goto out_unlock;
329 }
330
331 len = nla_len(info->attrs[IOAM6_ATTR_SC_DATA]);
332 len_aligned = ALIGN(len, 4);
333
334 sc = kzalloc(sizeof(*sc) + len_aligned, GFP_KERNEL);
335 if (!sc) {
336 err = -ENOMEM;
337 goto out_unlock;
338 }
339
340 sc->id = id;
341 sc->len = len_aligned;
342 sc->hdr = cpu_to_be32(sc->id | ((u8)(sc->len / 4) << 24));
343 nla_memcpy(sc->data, info->attrs[IOAM6_ATTR_SC_DATA], len);
344
345 err = rhashtable_lookup_insert_fast(&nsdata->schemas, &sc->head,
346 rht_sc_params);
347 if (err)
348 goto free_sc;
349
350 out_unlock:
351 mutex_unlock(&nsdata->lock);
352 return err;
353 free_sc:
354 kfree(sc);
355 goto out_unlock;
356 }
357
ioam6_genl_delsc(struct sk_buff * skb,struct genl_info * info)358 static int ioam6_genl_delsc(struct sk_buff *skb, struct genl_info *info)
359 {
360 struct ioam6_pernet_data *nsdata;
361 struct ioam6_namespace *ns;
362 struct ioam6_schema *sc;
363 int err;
364 u32 id;
365
366 if (!info->attrs[IOAM6_ATTR_SC_ID])
367 return -EINVAL;
368
369 id = nla_get_u32(info->attrs[IOAM6_ATTR_SC_ID]);
370 nsdata = ioam6_pernet(genl_info_net(info));
371
372 mutex_lock(&nsdata->lock);
373
374 sc = rhashtable_lookup_fast(&nsdata->schemas, &id, rht_sc_params);
375 if (!sc) {
376 err = -ENOENT;
377 goto out_unlock;
378 }
379
380 ns = rcu_dereference_protected(sc->ns, lockdep_is_held(&nsdata->lock));
381
382 err = rhashtable_remove_fast(&nsdata->schemas, &sc->head,
383 rht_sc_params);
384 if (err)
385 goto out_unlock;
386
387 if (ns)
388 rcu_assign_pointer(ns->schema, NULL);
389
390 ioam6_sc_release(sc);
391
392 out_unlock:
393 mutex_unlock(&nsdata->lock);
394 return err;
395 }
396
__ioam6_genl_dumpsc_element(struct ioam6_schema * sc,u32 portid,u32 seq,u32 flags,struct sk_buff * skb,u8 cmd)397 static int __ioam6_genl_dumpsc_element(struct ioam6_schema *sc,
398 u32 portid, u32 seq, u32 flags,
399 struct sk_buff *skb, u8 cmd)
400 {
401 struct ioam6_namespace *ns;
402 void *hdr;
403
404 hdr = genlmsg_put(skb, portid, seq, &ioam6_genl_family, flags, cmd);
405 if (!hdr)
406 return -ENOMEM;
407
408 if (nla_put_u32(skb, IOAM6_ATTR_SC_ID, sc->id) ||
409 nla_put(skb, IOAM6_ATTR_SC_DATA, sc->len, sc->data))
410 goto nla_put_failure;
411
412 rcu_read_lock();
413
414 ns = rcu_dereference(sc->ns);
415 if (ns && nla_put_u16(skb, IOAM6_ATTR_NS_ID, be16_to_cpu(ns->id))) {
416 rcu_read_unlock();
417 goto nla_put_failure;
418 }
419
420 rcu_read_unlock();
421
422 genlmsg_end(skb, hdr);
423 return 0;
424
425 nla_put_failure:
426 genlmsg_cancel(skb, hdr);
427 return -EMSGSIZE;
428 }
429
ioam6_genl_dumpsc_start(struct netlink_callback * cb)430 static int ioam6_genl_dumpsc_start(struct netlink_callback *cb)
431 {
432 struct ioam6_pernet_data *nsdata = ioam6_pernet(sock_net(cb->skb->sk));
433 struct rhashtable_iter *iter = (struct rhashtable_iter *)cb->args[0];
434
435 if (!iter) {
436 iter = kmalloc(sizeof(*iter), GFP_KERNEL);
437 if (!iter)
438 return -ENOMEM;
439
440 cb->args[0] = (long)iter;
441 }
442
443 rhashtable_walk_enter(&nsdata->schemas, iter);
444
445 return 0;
446 }
447
ioam6_genl_dumpsc_done(struct netlink_callback * cb)448 static int ioam6_genl_dumpsc_done(struct netlink_callback *cb)
449 {
450 struct rhashtable_iter *iter = (struct rhashtable_iter *)cb->args[0];
451
452 rhashtable_walk_exit(iter);
453 kfree(iter);
454
455 return 0;
456 }
457
ioam6_genl_dumpsc(struct sk_buff * skb,struct netlink_callback * cb)458 static int ioam6_genl_dumpsc(struct sk_buff *skb, struct netlink_callback *cb)
459 {
460 struct rhashtable_iter *iter;
461 struct ioam6_schema *sc;
462 int err;
463
464 iter = (struct rhashtable_iter *)cb->args[0];
465 rhashtable_walk_start(iter);
466
467 for (;;) {
468 sc = rhashtable_walk_next(iter);
469
470 if (IS_ERR(sc)) {
471 if (PTR_ERR(sc) == -EAGAIN)
472 continue;
473 err = PTR_ERR(sc);
474 goto done;
475 } else if (!sc) {
476 break;
477 }
478
479 err = __ioam6_genl_dumpsc_element(sc,
480 NETLINK_CB(cb->skb).portid,
481 cb->nlh->nlmsg_seq,
482 NLM_F_MULTI,
483 skb,
484 IOAM6_CMD_DUMP_SCHEMAS);
485 if (err)
486 goto done;
487 }
488
489 err = skb->len;
490
491 done:
492 rhashtable_walk_stop(iter);
493 return err;
494 }
495
ioam6_genl_ns_set_schema(struct sk_buff * skb,struct genl_info * info)496 static int ioam6_genl_ns_set_schema(struct sk_buff *skb, struct genl_info *info)
497 {
498 struct ioam6_namespace *ns, *ns_ref;
499 struct ioam6_schema *sc, *sc_ref;
500 struct ioam6_pernet_data *nsdata;
501 __be16 ns_id;
502 u32 sc_id;
503 int err;
504
505 if (!info->attrs[IOAM6_ATTR_NS_ID] ||
506 (!info->attrs[IOAM6_ATTR_SC_ID] &&
507 !info->attrs[IOAM6_ATTR_SC_NONE]))
508 return -EINVAL;
509
510 ns_id = cpu_to_be16(nla_get_u16(info->attrs[IOAM6_ATTR_NS_ID]));
511 nsdata = ioam6_pernet(genl_info_net(info));
512
513 mutex_lock(&nsdata->lock);
514
515 ns = rhashtable_lookup_fast(&nsdata->namespaces, &ns_id, rht_ns_params);
516 if (!ns) {
517 err = -ENOENT;
518 goto out_unlock;
519 }
520
521 if (info->attrs[IOAM6_ATTR_SC_NONE]) {
522 sc = NULL;
523 } else {
524 sc_id = nla_get_u32(info->attrs[IOAM6_ATTR_SC_ID]);
525 sc = rhashtable_lookup_fast(&nsdata->schemas, &sc_id,
526 rht_sc_params);
527 if (!sc) {
528 err = -ENOENT;
529 goto out_unlock;
530 }
531 }
532
533 sc_ref = rcu_dereference_protected(ns->schema,
534 lockdep_is_held(&nsdata->lock));
535 if (sc_ref)
536 rcu_assign_pointer(sc_ref->ns, NULL);
537 rcu_assign_pointer(ns->schema, sc);
538
539 if (sc) {
540 ns_ref = rcu_dereference_protected(sc->ns,
541 lockdep_is_held(&nsdata->lock));
542 if (ns_ref)
543 rcu_assign_pointer(ns_ref->schema, NULL);
544 rcu_assign_pointer(sc->ns, ns);
545 }
546
547 err = 0;
548
549 out_unlock:
550 mutex_unlock(&nsdata->lock);
551 return err;
552 }
553
554 static const struct genl_ops ioam6_genl_ops[] = {
555 {
556 .cmd = IOAM6_CMD_ADD_NAMESPACE,
557 .validate = GENL_DONT_VALIDATE_STRICT | GENL_DONT_VALIDATE_DUMP,
558 .doit = ioam6_genl_addns,
559 .flags = GENL_ADMIN_PERM,
560 .policy = ioam6_genl_policy_addns,
561 .maxattr = ARRAY_SIZE(ioam6_genl_policy_addns) - 1,
562 },
563 {
564 .cmd = IOAM6_CMD_DEL_NAMESPACE,
565 .validate = GENL_DONT_VALIDATE_STRICT | GENL_DONT_VALIDATE_DUMP,
566 .doit = ioam6_genl_delns,
567 .flags = GENL_ADMIN_PERM,
568 .policy = ioam6_genl_policy_delns,
569 .maxattr = ARRAY_SIZE(ioam6_genl_policy_delns) - 1,
570 },
571 {
572 .cmd = IOAM6_CMD_DUMP_NAMESPACES,
573 .validate = GENL_DONT_VALIDATE_STRICT | GENL_DONT_VALIDATE_DUMP,
574 .start = ioam6_genl_dumpns_start,
575 .dumpit = ioam6_genl_dumpns,
576 .done = ioam6_genl_dumpns_done,
577 .flags = GENL_ADMIN_PERM,
578 },
579 {
580 .cmd = IOAM6_CMD_ADD_SCHEMA,
581 .validate = GENL_DONT_VALIDATE_STRICT | GENL_DONT_VALIDATE_DUMP,
582 .doit = ioam6_genl_addsc,
583 .flags = GENL_ADMIN_PERM,
584 .policy = ioam6_genl_policy_addsc,
585 .maxattr = ARRAY_SIZE(ioam6_genl_policy_addsc) - 1,
586 },
587 {
588 .cmd = IOAM6_CMD_DEL_SCHEMA,
589 .validate = GENL_DONT_VALIDATE_STRICT | GENL_DONT_VALIDATE_DUMP,
590 .doit = ioam6_genl_delsc,
591 .flags = GENL_ADMIN_PERM,
592 .policy = ioam6_genl_policy_delsc,
593 .maxattr = ARRAY_SIZE(ioam6_genl_policy_delsc) - 1,
594 },
595 {
596 .cmd = IOAM6_CMD_DUMP_SCHEMAS,
597 .validate = GENL_DONT_VALIDATE_STRICT | GENL_DONT_VALIDATE_DUMP,
598 .start = ioam6_genl_dumpsc_start,
599 .dumpit = ioam6_genl_dumpsc,
600 .done = ioam6_genl_dumpsc_done,
601 .flags = GENL_ADMIN_PERM,
602 },
603 {
604 .cmd = IOAM6_CMD_NS_SET_SCHEMA,
605 .validate = GENL_DONT_VALIDATE_STRICT | GENL_DONT_VALIDATE_DUMP,
606 .doit = ioam6_genl_ns_set_schema,
607 .flags = GENL_ADMIN_PERM,
608 .policy = ioam6_genl_policy_ns_sc,
609 .maxattr = ARRAY_SIZE(ioam6_genl_policy_ns_sc) - 1,
610 },
611 };
612
613 static struct genl_family ioam6_genl_family __ro_after_init = {
614 .name = IOAM6_GENL_NAME,
615 .version = IOAM6_GENL_VERSION,
616 .netnsok = true,
617 .parallel_ops = true,
618 .ops = ioam6_genl_ops,
619 .n_ops = ARRAY_SIZE(ioam6_genl_ops),
620 .module = THIS_MODULE,
621 };
622
ioam6_namespace(struct net * net,__be16 id)623 struct ioam6_namespace *ioam6_namespace(struct net *net, __be16 id)
624 {
625 struct ioam6_pernet_data *nsdata = ioam6_pernet(net);
626
627 return rhashtable_lookup_fast(&nsdata->namespaces, &id, rht_ns_params);
628 }
629
__ioam6_fill_trace_data(struct sk_buff * skb,struct ioam6_namespace * ns,struct ioam6_trace_hdr * trace,struct ioam6_schema * sc,u8 sclen)630 static void __ioam6_fill_trace_data(struct sk_buff *skb,
631 struct ioam6_namespace *ns,
632 struct ioam6_trace_hdr *trace,
633 struct ioam6_schema *sc,
634 u8 sclen)
635 {
636 struct __kernel_sock_timeval ts;
637 u64 raw64;
638 u32 raw32;
639 u16 raw16;
640 u8 *data;
641 u8 byte;
642
643 data = trace->data + trace->remlen * 4 - trace->nodelen * 4 - sclen * 4;
644
645 /* hop_lim and node_id */
646 if (trace->type.bit0) {
647 byte = ipv6_hdr(skb)->hop_limit;
648 if (skb->dev)
649 byte--;
650
651 raw32 = dev_net(skb_dst(skb)->dev)->ipv6.sysctl.ioam6_id;
652
653 *(__be32 *)data = cpu_to_be32((byte << 24) | raw32);
654 data += sizeof(__be32);
655 }
656
657 /* ingress_if_id and egress_if_id */
658 if (trace->type.bit1) {
659 if (!skb->dev)
660 raw16 = IOAM6_U16_UNAVAILABLE;
661 else
662 raw16 = (__force u16)__in6_dev_get(skb->dev)->cnf.ioam6_id;
663
664 *(__be16 *)data = cpu_to_be16(raw16);
665 data += sizeof(__be16);
666
667 if (skb_dst(skb)->dev->flags & IFF_LOOPBACK)
668 raw16 = IOAM6_U16_UNAVAILABLE;
669 else
670 raw16 = (__force u16)__in6_dev_get(skb_dst(skb)->dev)->cnf.ioam6_id;
671
672 *(__be16 *)data = cpu_to_be16(raw16);
673 data += sizeof(__be16);
674 }
675
676 /* timestamp seconds */
677 if (trace->type.bit2) {
678 if (!skb->dev) {
679 *(__be32 *)data = cpu_to_be32(IOAM6_U32_UNAVAILABLE);
680 } else {
681 if (!skb->tstamp)
682 __net_timestamp(skb);
683
684 skb_get_new_timestamp(skb, &ts);
685 *(__be32 *)data = cpu_to_be32((u32)ts.tv_sec);
686 }
687 data += sizeof(__be32);
688 }
689
690 /* timestamp subseconds */
691 if (trace->type.bit3) {
692 if (!skb->dev) {
693 *(__be32 *)data = cpu_to_be32(IOAM6_U32_UNAVAILABLE);
694 } else {
695 if (!skb->tstamp)
696 __net_timestamp(skb);
697
698 if (!trace->type.bit2)
699 skb_get_new_timestamp(skb, &ts);
700
701 *(__be32 *)data = cpu_to_be32((u32)ts.tv_usec);
702 }
703 data += sizeof(__be32);
704 }
705
706 /* transit delay */
707 if (trace->type.bit4) {
708 *(__be32 *)data = cpu_to_be32(IOAM6_U32_UNAVAILABLE);
709 data += sizeof(__be32);
710 }
711
712 /* namespace data */
713 if (trace->type.bit5) {
714 *(__be32 *)data = ns->data;
715 data += sizeof(__be32);
716 }
717
718 /* queue depth */
719 if (trace->type.bit6) {
720 *(__be32 *)data = cpu_to_be32(IOAM6_U32_UNAVAILABLE);
721 data += sizeof(__be32);
722 }
723
724 /* checksum complement */
725 if (trace->type.bit7) {
726 *(__be32 *)data = cpu_to_be32(IOAM6_U32_UNAVAILABLE);
727 data += sizeof(__be32);
728 }
729
730 /* hop_lim and node_id (wide) */
731 if (trace->type.bit8) {
732 byte = ipv6_hdr(skb)->hop_limit;
733 if (skb->dev)
734 byte--;
735
736 raw64 = dev_net(skb_dst(skb)->dev)->ipv6.sysctl.ioam6_id_wide;
737
738 *(__be64 *)data = cpu_to_be64(((u64)byte << 56) | raw64);
739 data += sizeof(__be64);
740 }
741
742 /* ingress_if_id and egress_if_id (wide) */
743 if (trace->type.bit9) {
744 if (!skb->dev)
745 raw32 = IOAM6_U32_UNAVAILABLE;
746 else
747 raw32 = __in6_dev_get(skb->dev)->cnf.ioam6_id_wide;
748
749 *(__be32 *)data = cpu_to_be32(raw32);
750 data += sizeof(__be32);
751
752 if (skb_dst(skb)->dev->flags & IFF_LOOPBACK)
753 raw32 = IOAM6_U32_UNAVAILABLE;
754 else
755 raw32 = __in6_dev_get(skb_dst(skb)->dev)->cnf.ioam6_id_wide;
756
757 *(__be32 *)data = cpu_to_be32(raw32);
758 data += sizeof(__be32);
759 }
760
761 /* namespace data (wide) */
762 if (trace->type.bit10) {
763 *(__be64 *)data = ns->data_wide;
764 data += sizeof(__be64);
765 }
766
767 /* buffer occupancy */
768 if (trace->type.bit11) {
769 *(__be32 *)data = cpu_to_be32(IOAM6_U32_UNAVAILABLE);
770 data += sizeof(__be32);
771 }
772
773 /* bit12 undefined: filled with empty value */
774 if (trace->type.bit12) {
775 *(__be32 *)data = cpu_to_be32(IOAM6_U32_UNAVAILABLE);
776 data += sizeof(__be32);
777 }
778
779 /* bit13 undefined: filled with empty value */
780 if (trace->type.bit13) {
781 *(__be32 *)data = cpu_to_be32(IOAM6_U32_UNAVAILABLE);
782 data += sizeof(__be32);
783 }
784
785 /* bit14 undefined: filled with empty value */
786 if (trace->type.bit14) {
787 *(__be32 *)data = cpu_to_be32(IOAM6_U32_UNAVAILABLE);
788 data += sizeof(__be32);
789 }
790
791 /* bit15 undefined: filled with empty value */
792 if (trace->type.bit15) {
793 *(__be32 *)data = cpu_to_be32(IOAM6_U32_UNAVAILABLE);
794 data += sizeof(__be32);
795 }
796
797 /* bit16 undefined: filled with empty value */
798 if (trace->type.bit16) {
799 *(__be32 *)data = cpu_to_be32(IOAM6_U32_UNAVAILABLE);
800 data += sizeof(__be32);
801 }
802
803 /* bit17 undefined: filled with empty value */
804 if (trace->type.bit17) {
805 *(__be32 *)data = cpu_to_be32(IOAM6_U32_UNAVAILABLE);
806 data += sizeof(__be32);
807 }
808
809 /* bit18 undefined: filled with empty value */
810 if (trace->type.bit18) {
811 *(__be32 *)data = cpu_to_be32(IOAM6_U32_UNAVAILABLE);
812 data += sizeof(__be32);
813 }
814
815 /* bit19 undefined: filled with empty value */
816 if (trace->type.bit19) {
817 *(__be32 *)data = cpu_to_be32(IOAM6_U32_UNAVAILABLE);
818 data += sizeof(__be32);
819 }
820
821 /* bit20 undefined: filled with empty value */
822 if (trace->type.bit20) {
823 *(__be32 *)data = cpu_to_be32(IOAM6_U32_UNAVAILABLE);
824 data += sizeof(__be32);
825 }
826
827 /* bit21 undefined: filled with empty value */
828 if (trace->type.bit21) {
829 *(__be32 *)data = cpu_to_be32(IOAM6_U32_UNAVAILABLE);
830 data += sizeof(__be32);
831 }
832
833 /* opaque state snapshot */
834 if (trace->type.bit22) {
835 if (!sc) {
836 *(__be32 *)data = cpu_to_be32(IOAM6_U32_UNAVAILABLE >> 8);
837 } else {
838 *(__be32 *)data = sc->hdr;
839 data += sizeof(__be32);
840
841 memcpy(data, sc->data, sc->len);
842 }
843 }
844 }
845
846 /* called with rcu_read_lock() */
ioam6_fill_trace_data(struct sk_buff * skb,struct ioam6_namespace * ns,struct ioam6_trace_hdr * trace)847 void ioam6_fill_trace_data(struct sk_buff *skb,
848 struct ioam6_namespace *ns,
849 struct ioam6_trace_hdr *trace)
850 {
851 struct ioam6_schema *sc;
852 u8 sclen = 0;
853
854 /* Skip if Overflow flag is set
855 */
856 if (trace->overflow)
857 return;
858
859 /* NodeLen does not include Opaque State Snapshot length. We need to
860 * take it into account if the corresponding bit is set (bit 22) and
861 * if the current IOAM namespace has an active schema attached to it
862 */
863 sc = rcu_dereference(ns->schema);
864 if (trace->type.bit22) {
865 sclen = sizeof_field(struct ioam6_schema, hdr) / 4;
866
867 if (sc)
868 sclen += sc->len / 4;
869 }
870
871 /* If there is no space remaining, we set the Overflow flag and we
872 * skip without filling the trace
873 */
874 if (!trace->remlen || trace->remlen < trace->nodelen + sclen) {
875 trace->overflow = 1;
876 return;
877 }
878
879 __ioam6_fill_trace_data(skb, ns, trace, sc, sclen);
880 trace->remlen -= trace->nodelen + sclen;
881 }
882
ioam6_net_init(struct net * net)883 static int __net_init ioam6_net_init(struct net *net)
884 {
885 struct ioam6_pernet_data *nsdata;
886 int err = -ENOMEM;
887
888 nsdata = kzalloc(sizeof(*nsdata), GFP_KERNEL);
889 if (!nsdata)
890 goto out;
891
892 mutex_init(&nsdata->lock);
893 net->ipv6.ioam6_data = nsdata;
894
895 err = rhashtable_init(&nsdata->namespaces, &rht_ns_params);
896 if (err)
897 goto free_nsdata;
898
899 err = rhashtable_init(&nsdata->schemas, &rht_sc_params);
900 if (err)
901 goto free_rht_ns;
902
903 out:
904 return err;
905 free_rht_ns:
906 rhashtable_destroy(&nsdata->namespaces);
907 free_nsdata:
908 kfree(nsdata);
909 net->ipv6.ioam6_data = NULL;
910 goto out;
911 }
912
ioam6_net_exit(struct net * net)913 static void __net_exit ioam6_net_exit(struct net *net)
914 {
915 struct ioam6_pernet_data *nsdata = ioam6_pernet(net);
916
917 rhashtable_free_and_destroy(&nsdata->namespaces, ioam6_free_ns, NULL);
918 rhashtable_free_and_destroy(&nsdata->schemas, ioam6_free_sc, NULL);
919
920 kfree(nsdata);
921 }
922
923 static struct pernet_operations ioam6_net_ops = {
924 .init = ioam6_net_init,
925 .exit = ioam6_net_exit,
926 };
927
ioam6_init(void)928 int __init ioam6_init(void)
929 {
930 int err = register_pernet_subsys(&ioam6_net_ops);
931 if (err)
932 goto out;
933
934 err = genl_register_family(&ioam6_genl_family);
935 if (err)
936 goto out_unregister_pernet_subsys;
937
938 #ifdef CONFIG_IPV6_IOAM6_LWTUNNEL
939 err = ioam6_iptunnel_init();
940 if (err)
941 goto out_unregister_genl;
942 #endif
943
944 pr_info("In-situ OAM (IOAM) with IPv6\n");
945
946 out:
947 return err;
948 #ifdef CONFIG_IPV6_IOAM6_LWTUNNEL
949 out_unregister_genl:
950 genl_unregister_family(&ioam6_genl_family);
951 #endif
952 out_unregister_pernet_subsys:
953 unregister_pernet_subsys(&ioam6_net_ops);
954 goto out;
955 }
956
ioam6_exit(void)957 void ioam6_exit(void)
958 {
959 #ifdef CONFIG_IPV6_IOAM6_LWTUNNEL
960 ioam6_iptunnel_exit();
961 #endif
962 genl_unregister_family(&ioam6_genl_family);
963 unregister_pernet_subsys(&ioam6_net_ops);
964 }
965