Skip to content

Commit 338f665

Browse files
edumazetgregkh
authored andcommitted
ipv4: add reference counting to metrics
[ Upstream commit 3fb07daff8e99243366a081e5129560734de4ada ] Andrey Konovalov reported crashes in ipv4_mtu() I could reproduce the issue with KASAN kernels, between 10.246.7.151 and 10.246.7.152 : 1) 20 concurrent netperf -t TCP_RR -H 10.246.7.152 -l 1000 & 2) At the same time run following loop : while : do ip ro add 10.246.7.152 dev eth0 src 10.246.7.151 mtu 1500 ip ro del 10.246.7.152 dev eth0 src 10.246.7.151 mtu 1500 done Cong Wang attempted to add back rt->fi in commit 82486aa6f1b9 ("ipv4: restore rt->fi for reference counting") but this proved to add some issues that were complex to solve. Instead, I suggested to add a refcount to the metrics themselves, being a standalone object (in particular, no reference to other objects) I tried to make this patch as small as possible to ease its backport, instead of being super clean. Note that we believe that only ipv4 dst need to take care of the metric refcount. But if this is wrong, this patch adds the basic infrastructure to extend this to other families. Many thanks to Julian Anastasov for reviewing this patch, and Cong Wang for his efforts on this problem. Fixes: 2860583 ("ipv4: Kill rt->fi") Signed-off-by: Eric Dumazet <edumazet@google.com> Reported-by: Andrey Konovalov <andreyknvl@google.com> Reviewed-by: Julian Anastasov <ja@ssi.bg> Acked-by: Cong Wang <xiyou.wangcong@gmail.com> Signed-off-by: David S. Miller <davem@davemloft.net> Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
1 parent 97f5457 commit 338f665

5 files changed

Lines changed: 45 additions & 23 deletions

File tree

include/net/dst.h

Lines changed: 7 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -110,10 +110,16 @@ struct dst_entry {
110110
};
111111
};
112112

113+
struct dst_metrics {
114+
u32 metrics[RTAX_MAX];
115+
atomic_t refcnt;
116+
};
117+
extern const struct dst_metrics dst_default_metrics;
118+
113119
u32 *dst_cow_metrics_generic(struct dst_entry *dst, unsigned long old);
114-
extern const u32 dst_default_metrics[];
115120

116121
#define DST_METRICS_READ_ONLY 0x1UL
122+
#define DST_METRICS_REFCOUNTED 0x2UL
117123
#define DST_METRICS_FLAGS 0x3UL
118124
#define __DST_METRICS_PTR(Y) \
119125
((u32 *)((Y) & ~DST_METRICS_FLAGS))

include/net/ip_fib.h

Lines changed: 5 additions & 5 deletions
Original file line numberDiff line numberDiff line change
@@ -112,11 +112,11 @@ struct fib_info {
112112
unsigned char fib_type;
113113
__be32 fib_prefsrc;
114114
u32 fib_priority;
115-
u32 *fib_metrics;
116-
#define fib_mtu fib_metrics[RTAX_MTU-1]
117-
#define fib_window fib_metrics[RTAX_WINDOW-1]
118-
#define fib_rtt fib_metrics[RTAX_RTT-1]
119-
#define fib_advmss fib_metrics[RTAX_ADVMSS-1]
115+
struct dst_metrics *fib_metrics;
116+
#define fib_mtu fib_metrics->metrics[RTAX_MTU-1]
117+
#define fib_window fib_metrics->metrics[RTAX_WINDOW-1]
118+
#define fib_rtt fib_metrics->metrics[RTAX_RTT-1]
119+
#define fib_advmss fib_metrics->metrics[RTAX_ADVMSS-1]
120120
int fib_nhs;
121121
#ifdef CONFIG_IP_ROUTE_MULTIPATH
122122
int fib_weight;

net/core/dst.c

Lines changed: 14 additions & 9 deletions
Original file line numberDiff line numberDiff line change
@@ -151,13 +151,13 @@ int dst_discard_out(struct net *net, struct sock *sk, struct sk_buff *skb)
151151
}
152152
EXPORT_SYMBOL(dst_discard_out);
153153

154-
const u32 dst_default_metrics[RTAX_MAX + 1] = {
154+
const struct dst_metrics dst_default_metrics = {
155155
/* This initializer is needed to force linker to place this variable
156156
* into const section. Otherwise it might end into bss section.
157157
* We really want to avoid false sharing on this variable, and catch
158158
* any writes on it.
159159
*/
160-
[RTAX_MAX] = 0xdeadbeef,
160+
.refcnt = ATOMIC_INIT(1),
161161
};
162162

163163
void dst_init(struct dst_entry *dst, struct dst_ops *ops,
@@ -169,7 +169,7 @@ void dst_init(struct dst_entry *dst, struct dst_ops *ops,
169169
if (dev)
170170
dev_hold(dev);
171171
dst->ops = ops;
172-
dst_init_metrics(dst, dst_default_metrics, true);
172+
dst_init_metrics(dst, dst_default_metrics.metrics, true);
173173
dst->expires = 0UL;
174174
dst->path = dst;
175175
dst->from = NULL;
@@ -315,25 +315,30 @@ EXPORT_SYMBOL(dst_release);
315315

316316
u32 *dst_cow_metrics_generic(struct dst_entry *dst, unsigned long old)
317317
{
318-
u32 *p = kmalloc(sizeof(u32) * RTAX_MAX, GFP_ATOMIC);
318+
struct dst_metrics *p = kmalloc(sizeof(*p), GFP_ATOMIC);
319319

320320
if (p) {
321-
u32 *old_p = __DST_METRICS_PTR(old);
321+
struct dst_metrics *old_p = (struct dst_metrics *)__DST_METRICS_PTR(old);
322322
unsigned long prev, new;
323323

324-
memcpy(p, old_p, sizeof(u32) * RTAX_MAX);
324+
atomic_set(&p->refcnt, 1);
325+
memcpy(p->metrics, old_p->metrics, sizeof(p->metrics));
325326

326327
new = (unsigned long) p;
327328
prev = cmpxchg(&dst->_metrics, old, new);
328329

329330
if (prev != old) {
330331
kfree(p);
331-
p = __DST_METRICS_PTR(prev);
332+
p = (struct dst_metrics *)__DST_METRICS_PTR(prev);
332333
if (prev & DST_METRICS_READ_ONLY)
333334
p = NULL;
335+
} else if (prev & DST_METRICS_REFCOUNTED) {
336+
if (atomic_dec_and_test(&old_p->refcnt))
337+
kfree(old_p);
334338
}
335339
}
336-
return p;
340+
BUILD_BUG_ON(offsetof(struct dst_metrics, metrics) != 0);
341+
return (u32 *)p;
337342
}
338343
EXPORT_SYMBOL(dst_cow_metrics_generic);
339344

@@ -342,7 +347,7 @@ void __dst_destroy_metrics_generic(struct dst_entry *dst, unsigned long old)
342347
{
343348
unsigned long prev, new;
344349

345-
new = ((unsigned long) dst_default_metrics) | DST_METRICS_READ_ONLY;
350+
new = ((unsigned long) &dst_default_metrics) | DST_METRICS_READ_ONLY;
346351
prev = cmpxchg(&dst->_metrics, old, new);
347352
if (prev == old)
348353
kfree(__DST_METRICS_PTR(old));

net/ipv4/fib_semantics.c

Lines changed: 10 additions & 7 deletions
Original file line numberDiff line numberDiff line change
@@ -204,6 +204,7 @@ static void rt_fibinfo_free_cpus(struct rtable __rcu * __percpu *rtp)
204204
static void free_fib_info_rcu(struct rcu_head *head)
205205
{
206206
struct fib_info *fi = container_of(head, struct fib_info, rcu);
207+
struct dst_metrics *m;
207208

208209
change_nexthops(fi) {
209210
if (nexthop_nh->nh_dev)
@@ -214,8 +215,9 @@ static void free_fib_info_rcu(struct rcu_head *head)
214215
rt_fibinfo_free(&nexthop_nh->nh_rth_input);
215216
} endfor_nexthops(fi);
216217

217-
if (fi->fib_metrics != (u32 *) dst_default_metrics)
218-
kfree(fi->fib_metrics);
218+
m = fi->fib_metrics;
219+
if (m != &dst_default_metrics && atomic_dec_and_test(&m->refcnt))
220+
kfree(m);
219221
kfree(fi);
220222
}
221223

@@ -982,11 +984,11 @@ fib_convert_metrics(struct fib_info *fi, const struct fib_config *cfg)
982984
val = 255;
983985
if (type == RTAX_FEATURES && (val & ~RTAX_FEATURE_MASK))
984986
return -EINVAL;
985-
fi->fib_metrics[type - 1] = val;
987+
fi->fib_metrics->metrics[type - 1] = val;
986988
}
987989

988990
if (ecn_ca)
989-
fi->fib_metrics[RTAX_FEATURES - 1] |= DST_FEATURE_ECN_CA;
991+
fi->fib_metrics->metrics[RTAX_FEATURES - 1] |= DST_FEATURE_ECN_CA;
990992

991993
return 0;
992994
}
@@ -1044,11 +1046,12 @@ struct fib_info *fib_create_info(struct fib_config *cfg)
10441046
goto failure;
10451047
fib_info_cnt++;
10461048
if (cfg->fc_mx) {
1047-
fi->fib_metrics = kzalloc(sizeof(u32) * RTAX_MAX, GFP_KERNEL);
1049+
fi->fib_metrics = kzalloc(sizeof(*fi->fib_metrics), GFP_KERNEL);
10481050
if (!fi->fib_metrics)
10491051
goto failure;
1052+
atomic_set(&fi->fib_metrics->refcnt, 1);
10501053
} else
1051-
fi->fib_metrics = (u32 *) dst_default_metrics;
1054+
fi->fib_metrics = (struct dst_metrics *)&dst_default_metrics;
10521055

10531056
fi->fib_net = net;
10541057
fi->fib_protocol = cfg->fc_protocol;
@@ -1251,7 +1254,7 @@ int fib_dump_info(struct sk_buff *skb, u32 portid, u32 seq, int event,
12511254
if (fi->fib_priority &&
12521255
nla_put_u32(skb, RTA_PRIORITY, fi->fib_priority))
12531256
goto nla_put_failure;
1254-
if (rtnetlink_put_metrics(skb, fi->fib_metrics) < 0)
1257+
if (rtnetlink_put_metrics(skb, fi->fib_metrics->metrics) < 0)
12551258
goto nla_put_failure;
12561259

12571260
if (fi->fib_prefsrc &&

net/ipv4/route.c

Lines changed: 9 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -1356,8 +1356,12 @@ static void rt_add_uncached_list(struct rtable *rt)
13561356

13571357
static void ipv4_dst_destroy(struct dst_entry *dst)
13581358
{
1359+
struct dst_metrics *p = (struct dst_metrics *)DST_METRICS_PTR(dst);
13591360
struct rtable *rt = (struct rtable *) dst;
13601361

1362+
if (p != &dst_default_metrics && atomic_dec_and_test(&p->refcnt))
1363+
kfree(p);
1364+
13611365
if (!list_empty(&rt->rt_uncached)) {
13621366
struct uncached_list *ul = rt->rt_uncached_list;
13631367

@@ -1409,7 +1413,11 @@ static void rt_set_nexthop(struct rtable *rt, __be32 daddr,
14091413
rt->rt_gateway = nh->nh_gw;
14101414
rt->rt_uses_gateway = 1;
14111415
}
1412-
dst_init_metrics(&rt->dst, fi->fib_metrics, true);
1416+
dst_init_metrics(&rt->dst, fi->fib_metrics->metrics, true);
1417+
if (fi->fib_metrics != &dst_default_metrics) {
1418+
rt->dst._metrics |= DST_METRICS_REFCOUNTED;
1419+
atomic_inc(&fi->fib_metrics->refcnt);
1420+
}
14131421
#ifdef CONFIG_IP_ROUTE_CLASSID
14141422
rt->dst.tclassid = nh->nh_tclassid;
14151423
#endif

0 commit comments

Comments
 (0)