ila_common.c 2.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103
  1. #include <linux/errno.h>
  2. #include <linux/ip.h>
  3. #include <linux/kernel.h>
  4. #include <linux/module.h>
  5. #include <linux/skbuff.h>
  6. #include <linux/socket.h>
  7. #include <linux/types.h>
  8. #include <net/checksum.h>
  9. #include <net/ip.h>
  10. #include <net/ip6_fib.h>
  11. #include <net/lwtunnel.h>
  12. #include <net/protocol.h>
  13. #include <uapi/linux/ila.h>
  14. #include "ila.h"
  15. static __wsum get_csum_diff(struct ipv6hdr *ip6h, struct ila_params *p)
  16. {
  17. if (*(__be64 *)&ip6h->daddr == p->locator_match)
  18. return p->csum_diff;
  19. else
  20. return compute_csum_diff8((__be32 *)&ip6h->daddr,
  21. (__be32 *)&p->locator);
  22. }
  23. void update_ipv6_locator(struct sk_buff *skb, struct ila_params *p)
  24. {
  25. __wsum diff;
  26. struct ipv6hdr *ip6h = ipv6_hdr(skb);
  27. size_t nhoff = sizeof(struct ipv6hdr);
  28. /* First update checksum */
  29. switch (ip6h->nexthdr) {
  30. case NEXTHDR_TCP:
  31. if (likely(pskb_may_pull(skb, nhoff + sizeof(struct tcphdr)))) {
  32. struct tcphdr *th = (struct tcphdr *)
  33. (skb_network_header(skb) + nhoff);
  34. diff = get_csum_diff(ip6h, p);
  35. inet_proto_csum_replace_by_diff(&th->check, skb,
  36. diff, true);
  37. }
  38. break;
  39. case NEXTHDR_UDP:
  40. if (likely(pskb_may_pull(skb, nhoff + sizeof(struct udphdr)))) {
  41. struct udphdr *uh = (struct udphdr *)
  42. (skb_network_header(skb) + nhoff);
  43. if (uh->check || skb->ip_summed == CHECKSUM_PARTIAL) {
  44. diff = get_csum_diff(ip6h, p);
  45. inet_proto_csum_replace_by_diff(&uh->check, skb,
  46. diff, true);
  47. if (!uh->check)
  48. uh->check = CSUM_MANGLED_0;
  49. }
  50. }
  51. break;
  52. case NEXTHDR_ICMP:
  53. if (likely(pskb_may_pull(skb,
  54. nhoff + sizeof(struct icmp6hdr)))) {
  55. struct icmp6hdr *ih = (struct icmp6hdr *)
  56. (skb_network_header(skb) + nhoff);
  57. diff = get_csum_diff(ip6h, p);
  58. inet_proto_csum_replace_by_diff(&ih->icmp6_cksum, skb,
  59. diff, true);
  60. }
  61. break;
  62. }
  63. /* Now change destination address */
  64. *(__be64 *)&ip6h->daddr = p->locator;
  65. }
  66. static int __init ila_init(void)
  67. {
  68. int ret;
  69. ret = ila_lwt_init();
  70. if (ret)
  71. goto fail_lwt;
  72. ret = ila_xlat_init();
  73. if (ret)
  74. goto fail_xlat;
  75. return 0;
  76. fail_xlat:
  77. ila_lwt_fini();
  78. fail_lwt:
  79. return ret;
  80. }
  81. static void __exit ila_fini(void)
  82. {
  83. ila_xlat_fini();
  84. ila_lwt_fini();
  85. }
  86. module_init(ila_init);
  87. module_exit(ila_fini);
  88. MODULE_AUTHOR("Tom Herbert <tom@herbertland.com>");
  89. MODULE_LICENSE("GPL");