xt_CHECKSUM.c 2.3 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091
  1. /* iptables module for the packet checksum mangling
  2. *
  3. * (C) 2002 by Harald Welte <laforge@netfilter.org>
  4. * (C) 2010 Red Hat, Inc.
  5. *
  6. * Author: Michael S. Tsirkin <mst@redhat.com>
  7. *
  8. * This program is free software; you can redistribute it and/or modify
  9. * it under the terms of the GNU General Public License version 2 as
  10. * published by the Free Software Foundation.
  11. */
  12. #define pr_fmt(fmt) KBUILD_MODNAME ": " fmt
  13. #include <linux/module.h>
  14. #include <linux/skbuff.h>
  15. #include <linux/netfilter/x_tables.h>
  16. #include <linux/netfilter/xt_CHECKSUM.h>
  17. #include <linux/netfilter_ipv4/ip_tables.h>
  18. #include <linux/netfilter_ipv6/ip6_tables.h>
  19. MODULE_LICENSE("GPL");
  20. MODULE_AUTHOR("Michael S. Tsirkin <mst@redhat.com>");
  21. MODULE_DESCRIPTION("Xtables: checksum modification");
  22. MODULE_ALIAS("ipt_CHECKSUM");
  23. MODULE_ALIAS("ip6t_CHECKSUM");
  24. static unsigned int
  25. checksum_tg(struct sk_buff *skb, const struct xt_action_param *par)
  26. {
  27. if (skb->ip_summed == CHECKSUM_PARTIAL && !skb_is_gso(skb))
  28. skb_checksum_help(skb);
  29. return XT_CONTINUE;
  30. }
  31. static int checksum_tg_check(const struct xt_tgchk_param *par)
  32. {
  33. const struct xt_CHECKSUM_info *einfo = par->targinfo;
  34. const struct ip6t_ip6 *i6 = par->entryinfo;
  35. const struct ipt_ip *i4 = par->entryinfo;
  36. if (einfo->operation & ~XT_CHECKSUM_OP_FILL) {
  37. pr_info_ratelimited("unsupported CHECKSUM operation %x\n",
  38. einfo->operation);
  39. return -EINVAL;
  40. }
  41. if (!einfo->operation)
  42. return -EINVAL;
  43. switch (par->family) {
  44. case NFPROTO_IPV4:
  45. if (i4->proto == IPPROTO_UDP &&
  46. (i4->invflags & XT_INV_PROTO) == 0)
  47. return 0;
  48. break;
  49. case NFPROTO_IPV6:
  50. if ((i6->flags & IP6T_F_PROTO) &&
  51. i6->proto == IPPROTO_UDP &&
  52. (i6->invflags & XT_INV_PROTO) == 0)
  53. return 0;
  54. break;
  55. }
  56. pr_warn_once("CHECKSUM should be avoided. If really needed, restrict with \"-p udp\" and only use in OUTPUT\n");
  57. return 0;
  58. }
  59. static struct xt_target checksum_tg_reg __read_mostly = {
  60. .name = "CHECKSUM",
  61. .family = NFPROTO_UNSPEC,
  62. .target = checksum_tg,
  63. .targetsize = sizeof(struct xt_CHECKSUM_info),
  64. .table = "mangle",
  65. .checkentry = checksum_tg_check,
  66. .me = THIS_MODULE,
  67. };
  68. static int __init checksum_tg_init(void)
  69. {
  70. return xt_register_target(&checksum_tg_reg);
  71. }
  72. static void __exit checksum_tg_exit(void)
  73. {
  74. xt_unregister_target(&checksum_tg_reg);
  75. }
  76. module_init(checksum_tg_init);
  77. module_exit(checksum_tg_exit);