Table of Contents
Linux netfilter nf_conntrack_expect
I got a very weird issue with a GNU/Linux firewall used to do NAT for about 40 VoIP phones using the SIP protocol. Very often the phone calls broke unexpectedly, sometimes the call just drop, other times the voice stream stops.
Problem: all ports in use
Looking at the syslog I found a clear clue as to what the problem was:
nf_ct_sip: dropping packet: all ports in use for SDP media IN= OUT= SRC=a.a.a.a DST=b.b.b.b LEN=820 TOS=0x08 PREC=0x80 TTL=127 ID=23404 PROTO=UDP SPT=13182 DPT=5060 LEN=800 nf_ct_sip: dropping packet: cannot add expectation for voice IN= OUT= SRC=a.a.a.a DST=b.b.b.b LEN=820 TOS=0x08 PREC=0x80 TTL=127 ID=23404 PROTO=UDP SPT=13182 DPT=5060 LEN=800 nf_ct_sip: dropping packet: all ports in use for SDP media IN= OUT= SRC=a.a.a.a DST=b.b.b.b LEN=820 TOS=0x08 PREC=0x80 TTL=127 ID=23405 PROTO=UDP SPT=13182 DPT=5060 LEN=800 nf_ct_sip: dropping packet: cannot add expectation for voice IN= OUT= SRC=a.a.a.a DST=b.b.b.b LEN=820 TOS=0x08 PREC=0x80 TTL=127 ID=23405 PROTO=UDP SPT=13182 DPT=5060 LEN=800
So this is not a pure NAT problem, e.g. it is not a problem like a full NAT table. Here the problem is about the expect table, i.e. a table used by some specific helper modules to do connection tracking for specific network protocols. The specific connection tracking in my case is about SIP protocol, handled by kernel module nf_ct_sip.
The SIP protocol does not relay on normal TCP/IP sessions, so the association between the host behind NAT and the public server is not maintained by the normal NAT table. Instead, the kernel maintains the expect table, which can be inspected with:
cat /proc/net/nf_conntrack_expect 34 l3proto = 2 proto=17 src=b.b.b.b dst=a.a.a.a sport=0 dport=48968 PERMANENT sip/signalling 37 l3proto = 2 proto=17 src=b.b.b.b dst=a.a.a.a sport=0 dport=35460 PERMANENT sip/signalling 35 l3proto = 2 proto=17 src=b.b.b.b dst=a.a.a.a sport=0 dport=40949 PERMANENT sip/signalling 56 l3proto = 2 proto=17 src=b.b.b.b dst=a.a.a.a sport=0 dport=14161 PERMANENT sip/signalling ...
This should be the same information shown by:
conntrack -L expect
If you just want the count of entries:
conntrack --count expect
Statistics about creation/deletion of entries can be obtained with:
conntrack --stats expect
If the kernel is configured with CONFIG_NF_CONNTRACK_PROCFS (which is indeed declared obsolete), there are several procfs entries that can be inspected.
The expect table have a size limit, that can be inspected with:
cat /proc/sys/net/netfilter/nf_conntrack_expect_max
This is not the number of the expect table entries, but an unspecified size; I suspect that you have to divide by 4 to obtain the hash table size. It seems that the default nf_conntrack_expect_max
on GNU/Linux kernel 5.10 is a very low 256 (maybe 64 entries).
Solution
The solution is to enlarge the expect table. You can do it at runtime, after the nf_conntrack module is loaded, writing to the /proc/ pseudo filesystem. Or you can pass the expect_hashsize parameter at module load time.
The last method is better, because you don't know when the kernel module is loaded during the boot process, so you risk to write the /proc/
filesystem too early, when the module is still not present. So the best solution is to add a /etc/modprobe.d/local.conf file with this line:
options nf_conntrack expect_hashsize=2048
The number that you pass to expect_hashsize results multiplied by 4 when you inspect the /proc/sys/net/netfilter/nf_conntrack_expect_max pseudo file.
sysctl and systemd
You might think of a solution by adding a file /etc/sysctl.d/local.conf with this line:
# This does not work, because module nf_conntrack is not loaded yet. net.netfilter.nf_conntrack_expect_max=8192
But the systemd unit systemd-sysctl.service may enforce that setting too early, when the kernel module is not loaded yet. The error message in syslog will be;
Couldn't write '8192' to 'net/netfilter/nf_conntrack_expect_max', ignoring: No such file or directory
The option works instead if, after the bootstrap, you run:
systemctl restart systemd-sysctl.service
Who add entries into the expect table
Entries into the conntrack expect table are added by the netfilter helpers, e.g. the sip helper must be addressed by an iptables rule in the raw table, using the relevant port (e.g. the 5060/UDP for the SIP protocol).
Here it is ho to verify that the connection tracking sip helper rule exists into the raw table:
iptables -t raw -L -n Chain PREROUTING (policy ACCEPT) target prot opt source destination ... CT udp -- 0.0.0.0/0 0.0.0.0/0 udp dpt:5060 CT helper sip Chain OUTPUT (policy ACCEPT) target prot opt source destination ... CT udp -- 0.0.0.0/0 0.0.0.0/0 udp dpt:5060 CT helper sip
If the rules are missing, you can add them using the following commands:
iptables -A PREROUTING -t raw -p udp --dport 5060 -j CT --helper sip iptables -A OUTPUT -t raw -p udp --dport 5060 -j CT --helper sip
Shorewall and helpers
The Shorewall firewall can be configured to install helpers for some particular protocols, e.g. FTP, IRC, SIP, …
The configuration file /etc/shorewall/shorewall.conf may contain the options AUTOHELPERS and HELPERS. The first option defaults to AUTOHELPERS=Yes and the second defaults to HELPERS=, but it is recommended to set it to AUTOHELPERS=No on modern kernels (>= 3.5) because it is not desiderable to have ruleset with automatically associated helpers for all the applications (the default when HELPERS is empty).
A possible configuration may be the following:
AUTOHELPERS=Yes HELPERS=ftp,sip
in this case helpers will be configured into the iptables raw table for tcp dpt:21
(FTP) and udp dpt:5060
(SIP).
But the recommended configuration is the following:
AUTOHELPERS=No HELPERS=ftp,sip
And be more specific with helpers in /etc/shorewall/rules, e.g.:
HELPER loc - udp 5060 ; helper=sip
This combination will create the following iptable rule into the raw table:
Chain PREROUTING (policy ACCEPT 12 packets, 792 bytes) pkts bytes target prot opt in out source destination 0 0 CT 17 -- lan0 * 0.0.0.0/0 0.0.0.0/0 udp dpt:5060 CT helper sip
Or you can add this into /etc/shorewall/conntrack:
CT:helper:sip:PO
In this case the helper is instantiated into the raw table in both PREROUTING and OUTPUT chains.
The default Debian 12 Bookworm configuration for Shorewall provides a conntrack file where helpers are enabled only if the Shorewall AUTOHELPERS option is enabled (in shorewall.conf
) and if the CT_TARGET iptables/netfilter capability is available (verify the output of shorewall show capabilities
).