secondary address promotion causes infinite loop -- it arranges
for ifa->ifa_next to point back to itself.

Problem is that 'prev_prom' and 'last_prim' might point at the same entry,
so 'last_sec' pointer must be obtained after prev_prom->next update.

Fixes: 2638eb8b50cf ("net: ipv4: provide __rcu annotation for ifa_list")
Reported-by: Ran Rozenstein <ra...@mellanox.com>
Reported-by: Tariq Toukan <tar...@mellanox.com>
Signed-off-by: Florian Westphal <f...@strlen.de>
---
 net/ipv4/devinet.c | 3 ++-
 1 file changed, 2 insertions(+), 1 deletion(-)

diff --git a/net/ipv4/devinet.c b/net/ipv4/devinet.c
index 7874303220c5..137d1892395d 100644
--- a/net/ipv4/devinet.c
+++ b/net/ipv4/devinet.c
@@ -428,8 +428,9 @@ static void __inet_del_ifa(struct in_device *in_dev,
                if (prev_prom) {
                        struct in_ifaddr *last_sec;
 
-                       last_sec = rtnl_dereference(last_prim->ifa_next);
                        rcu_assign_pointer(prev_prom->ifa_next, next_sec);
+
+                       last_sec = rtnl_dereference(last_prim->ifa_next);
                        rcu_assign_pointer(promote->ifa_next, last_sec);
                        rcu_assign_pointer(last_prim->ifa_next, promote);
                }
-- 
2.21.0

Reply via email to