Commit f94e63801ab2791ed64c409d0f751f6a0c953ead

Authored by Florian Westphal
Committed by Pablo Neira Ayuso
1 parent 7acfda539c

netfilter: conntrack: reset tcp maxwin on re-register

Doug Smythies says:
  Sometimes it is desirable to temporarily disable, or clear,
  the iptables rule set on a computer being controlled via a
  secure shell session (SSH). While unwise on an internet facing
  computer, I also do it often on non-internet accessible computers
  while testing. Recently, this has become problematic, with the
  SSH session being dropped upon re-load of the rule set.

The problem is that when all rules are deleted, conntrack hooks get
unregistered.

In case the rules are re-added later, its possible that tcp window
has moved far enough so that all packets are considered invalid (out of
window) until entry expires (which can take forever, default
established timeout is 5 days).

Fix this by clearing maxwin of existing tcp connections on register.

v2: don't touch entries on hook removal.
v3: remove obsolete expiry check.

Reported-by: Doug Smythies <dsmythies@telus.net>
Fixes: 4d3a57f23dec59 ("netfilter: conntrack: do not enable connection tracking unless needed")
Signed-off-by: Florian Westphal <fw@strlen.de>
Signed-off-by: Pablo Neira Ayuso <pablo@netfilter.org>

Showing 1 changed file with 26 additions and 0 deletions Side-by-side Diff

net/netfilter/nf_conntrack_proto.c
... ... @@ -776,9 +776,26 @@
776 776 };
777 777 #endif
778 778  
  779 +static int nf_ct_tcp_fixup(struct nf_conn *ct, void *_nfproto)
  780 +{
  781 + u8 nfproto = (unsigned long)_nfproto;
  782 +
  783 + if (nf_ct_l3num(ct) != nfproto)
  784 + return 0;
  785 +
  786 + if (nf_ct_protonum(ct) == IPPROTO_TCP &&
  787 + ct->proto.tcp.state == TCP_CONNTRACK_ESTABLISHED) {
  788 + ct->proto.tcp.seen[0].td_maxwin = 0;
  789 + ct->proto.tcp.seen[1].td_maxwin = 0;
  790 + }
  791 +
  792 + return 0;
  793 +}
  794 +
779 795 static int nf_ct_netns_do_get(struct net *net, u8 nfproto)
780 796 {
781 797 struct nf_conntrack_net *cnet = net_generic(net, nf_conntrack_net_id);
  798 + bool fixup_needed = false;
782 799 int err = 0;
783 800  
784 801 mutex_lock(&nf_ct_proto_mutex);
... ... @@ -798,6 +815,8 @@
798 815 ARRAY_SIZE(ipv4_conntrack_ops));
799 816 if (err)
800 817 cnet->users4 = 0;
  818 + else
  819 + fixup_needed = true;
801 820 break;
802 821 #if IS_ENABLED(CONFIG_IPV6)
803 822 case NFPROTO_IPV6:
... ... @@ -814,6 +833,8 @@
814 833 ARRAY_SIZE(ipv6_conntrack_ops));
815 834 if (err)
816 835 cnet->users6 = 0;
  836 + else
  837 + fixup_needed = true;
817 838 break;
818 839 #endif
819 840 default:
... ... @@ -822,6 +843,11 @@
822 843 }
823 844 out_unlock:
824 845 mutex_unlock(&nf_ct_proto_mutex);
  846 +
  847 + if (fixup_needed)
  848 + nf_ct_iterate_cleanup_net(net, nf_ct_tcp_fixup,
  849 + (void *)(unsigned long)nfproto, 0, 0);
  850 +
825 851 return err;
826 852 }
827 853