}
static inline int
- nl_allow_replace(struct krt_proto *p, rte *new)
-nl_allow_replace(struct krt_proto *p, rte *new, rte *old)
++nl_allow_replace(struct krt_proto *p, const rte *new, const rte *old)
{
- uint new_metric = ea_get_int(new->attrs->eattrs, EA_KRT_METRIC, 0);
- uint old_metric = ea_get_int(old->attrs->eattrs, EA_KRT_METRIC, 0);
+ /*
+ * In kernel routing tables, (net, metric) is the primary key. Therefore, we
+ * can use NL_OP_REPLACE only if the new and and the old route have the same
+ * kernel metric, otherwise the replace would just add the new route while
+ * keep the old one.
+ */
+
+ if ((p->af != AF_MPLS) && (KRT_CF->sys.metric == 0))
+ {
++ uint new_metric = ea_get_int(new->attrs, &ea_krt_metric, 0);
++ uint old_metric = ea_get_int(old->attrs, &ea_krt_metric, 0);
+
+ if (new_metric != old_metric)
+ return 0;
+ }
+
/*
* We use NL_OP_REPLACE for IPv4, it has an issue with not checking for
* matching rtm_protocol, but that is OK when dedicated priority is used.