Hard-coded M_* flags
YONETANI Tomokazu
qhwt+dfly at les.ath.cx
Wed Jun 7 20:45:22 PDT 2006
Hi, all.
I've been trying to update my PPPoE router/mail server in my room from
1.2.6-RELEASE to 1.4.4-RELEASE for a week now and always resulted
in a panic saying "bad mbuf flags" in m_free(). At first I thought
it was netgraph code was at fault because I'm using mpd, but the same
thing happened with ppp. After poking here and there in the source tree,
I found that ipfw2 is abusing mbuf mh_flags by defining M_SKIP_FIREWALL
as 0x4000 in ip_fw2.c file. Since 0x4000 in mh_flags also mean
M_EXT_CLUSTER, this produced an mbuf entry with M_EXT_CLUSTER set
without M_EXT, which led to a panic. Argh.
Instead of dumping ipfw and switching over to pf, I decided to investigate
other instances of M_* flags in DragonFly_RELEASE_1_4 source code by
grep'ping through with the following pattern:
$ egrep -r '(\.|->)mh?_flags *[|&]=? *[(~]?M_' /sys/
So far the problematic(IMO) flags are M_SKIP_FIREWALL, M_FEC_*,
and M_NOTIFICATION. I tried to think about how to solve the problem, but
as m_hdr.mh_flags is 16bit-wide, there's no extra room left to fit them.
(and changing the size of struct mbuf within a release branch is not
desirable, right?) I thought M_PROTO4 and M_PROTO5 can be used since
they are ipv6-specific and ipfw doesn't handle ipv6, but I don't know.
On FreeBSD 5 or later, M_SKIP_FIREWALL is already moved to <sys/mbuf.h>,
but M_FEC_* are left hardcoded in ng_fec.c.
[hardcoded M_ flags]
M_SKIP_FIREWALL: defined as 0x4000, which conflicts with M_EXT_CLUSTER
/sys/net/ipfw/ip_fw2.c
M_FEC_MAC: defined as 0x2000, which may conflict with M_CLCACHE
M_FEC_INET: defined as 0x4000, which may conflict with M_EXT_CLUSTER
M_FEC_INET6: defined as 0x8000, which may conflict with M_PHCACHE
/sys/netgraph/fec/ng_fec.c
(I've never used ng_fec, BTW)
M_NOTIFICATION: defined as 0x10000 in mbuf.h(doesn't fit in a short int!)
/sys/kern/uipc_socket.c
/sys/netinet/sctputil.c
M_CANFASTFWD: probably harmless, unless someone defines M_CANFASTFWD
/sys/contrib/ipfilter/netinet/fil.c
M_HASFCS: probably harmless, unless someone defines M_HASFCS
/sys/netproto/802_11/ieee80211_input.c
M_LINK0: probably harmless, unless someone defines M_LINK0
/sys/net/if_fddisubr.c
[uses of M_PROTOn flags]
M_PROTO1: defined in mbuf.h
/sys/dev/netif/bge/if_bge.c
/sys/dev/netif/em/if_em.c
/sys/dev/netif/gx/if_gx.c
/sys/dev/netif/nge/if_nge.c
/sys/dev/netif/re/if_re.c
/sys/dev/netif/ti/if_ti.c
/sys/dev/netif/txp/if_txp.c
/sys/net/bridge/if_bridge.c
/sys/net/if_ethersubr.c
/sys/net/vlan/if_vlan.c
M_AUTHIPHDR: defined as M_PROTO2 in in6.h
/sys/netinet6/ah_input.c
/sys/netinet6/in6.h
/sys/netinet6/ip6_input.c
/sys/netinet6/ipsec.c
/sys/netinet6/nd6_rtr.c
/sys/netproto/ipsec/ipsec.c
/sys/netproto/ipsec/xform_ah.c
/sys/netproto/ipsec/xform_ipip.c
M_HIGHPRI: defined as M_PROTO2 in if_ppp.c
/sys/net/ppp/if_ppp.c
M_DECRYPTED: defined as M_PROTO3 in in6.h
/sys/netinet/ip_icmp.c
/sys/netinet6/esp_input.c
/sys/netinet6/icmp6.c
/sys/netinet6/in6.h
/sys/netinet6/ipcomp_input.c
/sys/netinet6/ipsec.c
/sys/netproto/ipsec/xform_esp.c
/sys/netproto/ipsec/ipsec.c
/sys/netproto/ipsec/xform_ipip.c
M_ERRMARK: defined as M_PROTO3 in if_ppp.c
/sys/net/ppp/if_ppp.c
M_LOOP: defined as M_PROTO4 in ip6.h
/sys/net/if_loop.c (#ifdef INET6)
/sys/inetinet6/ip6.h
/sys/inetinet6/icmp6.c
/sys/inetinet6/ip6_input.c
/sys/inetinet6/ip6_mroute.c
/sys/inetinet6/ip6_output.c
/sys/inetinet6/mld6.c
M_AUTHIPDGM: defined as M_PROTO5 in in6.h
/sys/netinet6/ah_input.c
/sys/netinet6/esp_input.c
/sys/netinet6/in6.h
/sys/netinet6/ip6_input.c
/sys/netinet6/ipsec.c
/sys/netinet6/nd6_rtr.c
/sys/netproto/ipsec/ipsec.c
/sys/netproto/ipsec/xform_ah.c
/sys/netproto/ipsec/xform_ipip.c
More information about the Bugs
mailing list