icmpv6: Fix mbuf change in mld
After r354748 mld_input() can change the mbuf. The new pointer is never returned to icmp6_input() and when passed to icmp6_rip6_input() the mbuf may no longer valid leading to a panic. Pass a pointer to the mbuf to mld_input() so we can return an updated version in the non-error case. Add a test sending an MLD packet case which will trigger this bug. Pointyhat to: bz Reported by: gallatin, thj MFC After: 2 weeks X-MFC with: r354748 Sponsored by: Netflix
This commit is contained in:
parent
20241072c8
commit
32af08ecad
@ -604,7 +604,7 @@ icmp6_input(struct mbuf **mp, int *offp, int proto)
|
|||||||
*/
|
*/
|
||||||
if ((ip6->ip6_hlim != 1) || (m->m_flags & M_RTALERT_MLD) == 0)
|
if ((ip6->ip6_hlim != 1) || (m->m_flags & M_RTALERT_MLD) == 0)
|
||||||
goto freeit;
|
goto freeit;
|
||||||
if (mld_input(m, off, icmp6len) != 0) {
|
if (mld_input(&m, off, icmp6len) != 0) {
|
||||||
*mp = NULL;
|
*mp = NULL;
|
||||||
return (IPPROTO_DONE);
|
return (IPPROTO_DONE);
|
||||||
}
|
}
|
||||||
|
@ -1249,13 +1249,15 @@ mld_v1_input_report(struct ifnet *ifp, const struct ip6_hdr *ip6,
|
|||||||
* Return IPPROTO_DONE if we freed m. Otherwise, return 0.
|
* Return IPPROTO_DONE if we freed m. Otherwise, return 0.
|
||||||
*/
|
*/
|
||||||
int
|
int
|
||||||
mld_input(struct mbuf *m, int off, int icmp6len)
|
mld_input(struct mbuf **mp, int off, int icmp6len)
|
||||||
{
|
{
|
||||||
struct ifnet *ifp;
|
struct ifnet *ifp;
|
||||||
struct ip6_hdr *ip6;
|
struct ip6_hdr *ip6;
|
||||||
|
struct mbuf *m;
|
||||||
struct mld_hdr *mld;
|
struct mld_hdr *mld;
|
||||||
int mldlen;
|
int mldlen;
|
||||||
|
|
||||||
|
m = *mp;
|
||||||
CTR3(KTR_MLD, "%s: called w/mbuf (%p,%d)", __func__, m, off);
|
CTR3(KTR_MLD, "%s: called w/mbuf (%p,%d)", __func__, m, off);
|
||||||
|
|
||||||
ifp = m->m_pkthdr.rcvif;
|
ifp = m->m_pkthdr.rcvif;
|
||||||
@ -1278,6 +1280,7 @@ mld_input(struct mbuf *m, int off, int icmp6len)
|
|||||||
ICMP6STAT_INC(icp6s_badlen);
|
ICMP6STAT_INC(icp6s_badlen);
|
||||||
return (IPPROTO_DONE);
|
return (IPPROTO_DONE);
|
||||||
}
|
}
|
||||||
|
*mp = m;
|
||||||
ip6 = mtod(m, struct ip6_hdr *);
|
ip6 = mtod(m, struct ip6_hdr *);
|
||||||
mld = (struct mld_hdr *)(mtod(m, uint8_t *) + off);
|
mld = (struct mld_hdr *)(mtod(m, uint8_t *) + off);
|
||||||
|
|
||||||
|
@ -167,7 +167,7 @@ struct mld_ifsoftc *
|
|||||||
void mld_domifdetach(struct ifnet *);
|
void mld_domifdetach(struct ifnet *);
|
||||||
void mld_fasttimo(void);
|
void mld_fasttimo(void);
|
||||||
void mld_ifdetach(struct ifnet *, struct in6_multi_head *);
|
void mld_ifdetach(struct ifnet *, struct in6_multi_head *);
|
||||||
int mld_input(struct mbuf *, int, int);
|
int mld_input(struct mbuf **, int, int);
|
||||||
void mld_slowtimo(void);
|
void mld_slowtimo(void);
|
||||||
|
|
||||||
#ifdef SYSCTL_DECL
|
#ifdef SYSCTL_DECL
|
||||||
|
@ -7,12 +7,15 @@ FILESDIR= ${TESTSDIR}
|
|||||||
|
|
||||||
ATF_TESTS_SH= \
|
ATF_TESTS_SH= \
|
||||||
exthdr \
|
exthdr \
|
||||||
|
mld \
|
||||||
scapyi386
|
scapyi386
|
||||||
|
|
||||||
${PACKAGE}FILES+= exthdr.py
|
${PACKAGE}FILES+= exthdr.py
|
||||||
|
${PACKAGE}FILES+= mld.py
|
||||||
${PACKAGE}FILES+= scapyi386.py
|
${PACKAGE}FILES+= scapyi386.py
|
||||||
|
|
||||||
${PACKAGE}FILESMODE_exthdr.py= 0555
|
${PACKAGE}FILESMODE_exthdr.py= 0555
|
||||||
|
${PACKAGE}FILESMODE_mld.py= 0555
|
||||||
${PACKAGE}FILESMODE_scapyi386.py=0555
|
${PACKAGE}FILESMODE_scapyi386.py=0555
|
||||||
|
|
||||||
TESTS_SUBDIRS+= frag6
|
TESTS_SUBDIRS+= frag6
|
||||||
|
76
tests/sys/netinet6/mld.py
Normal file
76
tests/sys/netinet6/mld.py
Normal file
@ -0,0 +1,76 @@
|
|||||||
|
#!/usr/bin/env python
|
||||||
|
#-
|
||||||
|
# SPDX-License-Identifier: BSD-2-Clause
|
||||||
|
#
|
||||||
|
# Copyright (c) 2019 Netflix, Inc.
|
||||||
|
#
|
||||||
|
# Redistribution and use in source and binary forms, with or without
|
||||||
|
# modification, are permitted provided that the following conditions
|
||||||
|
# are met:
|
||||||
|
# 1. Redistributions of source code must retain the above copyright
|
||||||
|
# notice, this list of conditions and the following disclaimer.
|
||||||
|
# 2. Redistributions in binary form must reproduce the above copyright
|
||||||
|
# notice, this list of conditions and the following disclaimer in the
|
||||||
|
# documentation and/or other materials provided with the distribution.
|
||||||
|
#
|
||||||
|
# THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
|
||||||
|
# ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
||||||
|
# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
||||||
|
# ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
|
||||||
|
# FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
||||||
|
# DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
|
||||||
|
# OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
||||||
|
# HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
||||||
|
# LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
||||||
|
# OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
||||||
|
# SUCH DAMAGE.
|
||||||
|
#
|
||||||
|
# $FreeBSD$
|
||||||
|
#
|
||||||
|
|
||||||
|
import argparse
|
||||||
|
import scapy.all as sp
|
||||||
|
import socket
|
||||||
|
import sys
|
||||||
|
import binascii
|
||||||
|
|
||||||
|
def main():
|
||||||
|
parser = argparse.ArgumentParser("scapyi386.py",
|
||||||
|
description="IPv6 Ethernet Dest MAC test")
|
||||||
|
parser.add_argument('--sendif', nargs=1,
|
||||||
|
required=True,
|
||||||
|
help='The interface through which the packet will be sent')
|
||||||
|
parser.add_argument('--recvif', nargs=1,
|
||||||
|
required=True,
|
||||||
|
help='The interface on which to check for the packet')
|
||||||
|
parser.add_argument('--src', nargs=1,
|
||||||
|
required=True,
|
||||||
|
help='The source IP address')
|
||||||
|
parser.add_argument('--to', nargs=1,
|
||||||
|
required=True,
|
||||||
|
help='The destination IP address')
|
||||||
|
parser.add_argument('--debug',
|
||||||
|
required=False, action='store_true',
|
||||||
|
help='Enable test debugging')
|
||||||
|
parser.add_argument('--mldraw01',
|
||||||
|
required=False, action='store_true',
|
||||||
|
help='Multicast Listener Query Raw01')
|
||||||
|
|
||||||
|
args = parser.parse_args()
|
||||||
|
|
||||||
|
pkt = None
|
||||||
|
if args.mldraw01:
|
||||||
|
pkt = sp.Ether() / \
|
||||||
|
sp.IPv6(dst="ff02::1", hlim=1, nh=0) / \
|
||||||
|
sp.IPv6ExtHdrHopByHop(options = sp.RouterAlert(value=0)) / \
|
||||||
|
sp.ICMPv6MLQuery()
|
||||||
|
if pkt is None:
|
||||||
|
sys.exit(1)
|
||||||
|
if args.debug:
|
||||||
|
pkt.display()
|
||||||
|
sp.sendp(pkt, iface=args.sendif[0], verbose=False)
|
||||||
|
|
||||||
|
sys.exit(0)
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
main()
|
89
tests/sys/netinet6/mld.sh
Executable file
89
tests/sys/netinet6/mld.sh
Executable file
@ -0,0 +1,89 @@
|
|||||||
|
# $FreeBSD$
|
||||||
|
#-
|
||||||
|
# SPDX-License-Identifier: BSD-2-Clause
|
||||||
|
#
|
||||||
|
# Copyright (c) 2019 Netflix, Inc.
|
||||||
|
#
|
||||||
|
# Redistribution and use in source and binary forms, with or without
|
||||||
|
# modification, are permitted provided that the following conditions
|
||||||
|
# are met:
|
||||||
|
# 1. Redistributions of source code must retain the above copyright
|
||||||
|
# notice, this list of conditions and the following disclaimer.
|
||||||
|
# 2. Redistributions in binary form must reproduce the above copyright
|
||||||
|
# notice, this list of conditions and the following disclaimer in the
|
||||||
|
# documentation and/or other materials provided with the distribution.
|
||||||
|
#
|
||||||
|
# THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
|
||||||
|
# ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
||||||
|
# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
||||||
|
# ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
|
||||||
|
# FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
||||||
|
# DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
|
||||||
|
# OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
||||||
|
# HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
|
||||||
|
# LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
|
||||||
|
# OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
||||||
|
# SUCH DAMAGE.
|
||||||
|
#
|
||||||
|
|
||||||
|
. $(atf_get_srcdir)/../common/vnet.subr
|
||||||
|
|
||||||
|
atf_test_case "mldraw01" "cleanup"
|
||||||
|
mldraw01_head() {
|
||||||
|
|
||||||
|
atf_set descr 'Test for correct Ethernet Destination MAC address'
|
||||||
|
atf_set require.user root
|
||||||
|
atf_set require.progs scapy
|
||||||
|
}
|
||||||
|
|
||||||
|
mldraw01_body() {
|
||||||
|
|
||||||
|
ids=65533
|
||||||
|
id=`printf "%x" ${ids}`
|
||||||
|
if [ $$ -gt 65535 ]; then
|
||||||
|
xl=`printf "%x" $(($$ - 65535))`
|
||||||
|
yl="1"
|
||||||
|
else
|
||||||
|
xl=`printf "%x" $$`
|
||||||
|
yl=""
|
||||||
|
fi
|
||||||
|
|
||||||
|
vnet_init
|
||||||
|
|
||||||
|
ip6a="2001:db8:6666:0000:${yl}:${id}:1:${xl}"
|
||||||
|
ip6b="2001:db8:6666:0000:${yl}:${id}:2:${xl}"
|
||||||
|
|
||||||
|
epair=$(vnet_mkepair)
|
||||||
|
ifconfig ${epair}a up
|
||||||
|
ifconfig ${epair}a inet6 ${ip6a}/64
|
||||||
|
|
||||||
|
jname="v6t-${id}-${yl}-${xl}"
|
||||||
|
vnet_mkjail ${jname} ${epair}b
|
||||||
|
jexec ${jname} ifconfig ${epair}b up
|
||||||
|
jexec ${jname} ifconfig ${epair}b inet6 ${ip6b}/64
|
||||||
|
|
||||||
|
# Let IPv6 ND do its thing.
|
||||||
|
#ping6 -q -c 1 ff02::1%${epair}a
|
||||||
|
#ping6 -q -c 1 ${ip6b}
|
||||||
|
sleep 3
|
||||||
|
|
||||||
|
pyname=$(atf_get ident)
|
||||||
|
|
||||||
|
atf_check -s exit:0 $(atf_get_srcdir)/mld.py \
|
||||||
|
--sendif ${epair}a --recvif ${epair}a \
|
||||||
|
--src ${ip6a} --to ${ip6b} \
|
||||||
|
--${pyname}
|
||||||
|
}
|
||||||
|
|
||||||
|
mldraw01_cleanup() {
|
||||||
|
|
||||||
|
vnet_cleanup
|
||||||
|
}
|
||||||
|
|
||||||
|
atf_init_test_cases()
|
||||||
|
{
|
||||||
|
|
||||||
|
atf_add_test_case "mldraw01"
|
||||||
|
}
|
||||||
|
|
||||||
|
# end
|
Loading…
Reference in New Issue
Block a user