Module Name: src
Committed By: ozaki-r
Date: Mon Sep 25 04:15:33 UTC 2017
Modified Files:
src/sys/net: route.c
Log Message:
Synchronize on rtcache_generation with rtlock
It's racy if NET_MPSAFE is enabled.
Pointed out by joerg@
To generate a diff of this commit:
cvs rdiff -u -r1.200 -r1.201 src/sys/net/route.c
Please note that diffs are not public domain; they are subject to the
copyright notices on the relevant files.
Modified files:
Index: src/sys/net/route.c
diff -u src/sys/net/route.c:1.200 src/sys/net/route.c:1.201
--- src/sys/net/route.c:1.200 Fri Sep 22 05:05:32 2017
+++ src/sys/net/route.c Mon Sep 25 04:15:33 2017
@@ -1,4 +1,4 @@
-/* $NetBSD: route.c,v 1.200 2017/09/22 05:05:32 ozaki-r Exp $ */
+/* $NetBSD: route.c,v 1.201 2017/09/25 04:15:33 ozaki-r Exp $ */
/*-
* Copyright (c) 1998, 2008 The NetBSD Foundation, Inc.
@@ -97,7 +97,7 @@
#endif
#include <sys/cdefs.h>
-__KERNEL_RCSID(0, "$NetBSD: route.c,v 1.200 2017/09/22 05:05:32 ozaki-r Exp $");
+__KERNEL_RCSID(0, "$NetBSD: route.c,v 1.201 2017/09/25 04:15:33 ozaki-r Exp $");
#include <sys/param.h>
#ifdef RTFLUSH_DEBUG
@@ -484,6 +484,8 @@ static void
rtcache_invalidate(void)
{
+ RT_ASSERT_WLOCK();
+
if (rtcache_debug())
printf("%s: enter\n", __func__);
@@ -1199,10 +1201,10 @@ rtrequest1(int req, struct rt_addrinfo *
rt_ref(rt);
RT_REFCNT_TRACE(rt);
}
+ rtcache_invalidate();
RT_UNLOCK();
need_unlock = false;
rt_timer_remove_all(rt);
- rtcache_invalidate();
#if defined(INET) || defined(INET6)
if (netmask != NULL)
lltable_prefix_free(dst->sa_family, dst, netmask, 0);
@@ -1295,9 +1297,9 @@ rtrequest1(int req, struct rt_addrinfo *
rt_ref(rt);
RT_REFCNT_TRACE(rt);
}
+ rtcache_invalidate();
RT_UNLOCK();
need_unlock = false;
- rtcache_invalidate();
break;
case RTM_GET:
if (netmask != NULL) {
@@ -1898,8 +1900,10 @@ rtcache_copy(struct route *new_ro, struc
if (ret != 0)
goto out;
+ RT_RLOCK();
new_ro->_ro_rt = rt;
new_ro->ro_rtcache_generation = rtcache_generation;
+ RT_UNLOCK();
rtcache_invariants(new_ro);
out:
rtcache_unref(rt, old_ro);
@@ -1957,16 +1961,17 @@ rtcache_validate(struct route *ro)
retry:
#endif
rtcache_invariants(ro);
+ RT_RLOCK();
if (ro->ro_rtcache_generation != rtcache_generation) {
/* The cache is invalidated */
- return NULL;
+ rt = NULL;
+ goto out;
}
rt = ro->_ro_rt;
if (rt == NULL)
- return NULL;
+ goto out;
- RT_RLOCK();
if ((rt->rt_flags & RTF_UP) == 0) {
rt = NULL;
goto out;