net: Add routes to the table associated with the device
When a device associated with a VRF is brought up or down routes should be added to/removed from the table associated with the VRF. fib_magic defaults to using the main or local tables. Have it use the table with the device if there is one. A part of this is directing prefsrc validations to the correct table as well. Signed-off-by: David Ahern <dsa@cumulusnetworks.com> Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
Родитель
30bbaa1950
Коммит
021dd3b8a1
|
@ -800,6 +800,7 @@ out:
|
||||||
static void fib_magic(int cmd, int type, __be32 dst, int dst_len, struct in_ifaddr *ifa)
|
static void fib_magic(int cmd, int type, __be32 dst, int dst_len, struct in_ifaddr *ifa)
|
||||||
{
|
{
|
||||||
struct net *net = dev_net(ifa->ifa_dev->dev);
|
struct net *net = dev_net(ifa->ifa_dev->dev);
|
||||||
|
int tb_id = vrf_dev_table_rtnl(ifa->ifa_dev->dev);
|
||||||
struct fib_table *tb;
|
struct fib_table *tb;
|
||||||
struct fib_config cfg = {
|
struct fib_config cfg = {
|
||||||
.fc_protocol = RTPROT_KERNEL,
|
.fc_protocol = RTPROT_KERNEL,
|
||||||
|
@ -814,11 +815,10 @@ static void fib_magic(int cmd, int type, __be32 dst, int dst_len, struct in_ifad
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
if (type == RTN_UNICAST)
|
if (!tb_id)
|
||||||
tb = fib_new_table(net, RT_TABLE_MAIN);
|
tb_id = (type == RTN_UNICAST) ? RT_TABLE_MAIN : RT_TABLE_LOCAL;
|
||||||
else
|
|
||||||
tb = fib_new_table(net, RT_TABLE_LOCAL);
|
|
||||||
|
|
||||||
|
tb = fib_new_table(net, tb_id);
|
||||||
if (!tb)
|
if (!tb)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
|
|
@ -838,6 +838,23 @@ __be32 fib_info_update_nh_saddr(struct net *net, struct fib_nh *nh)
|
||||||
return nh->nh_saddr;
|
return nh->nh_saddr;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static bool fib_valid_prefsrc(struct fib_config *cfg, __be32 fib_prefsrc)
|
||||||
|
{
|
||||||
|
if (cfg->fc_type != RTN_LOCAL || !cfg->fc_dst ||
|
||||||
|
fib_prefsrc != cfg->fc_dst) {
|
||||||
|
int tb_id = cfg->fc_table;
|
||||||
|
|
||||||
|
if (tb_id == RT_TABLE_MAIN)
|
||||||
|
tb_id = RT_TABLE_LOCAL;
|
||||||
|
|
||||||
|
if (inet_addr_type_table(cfg->fc_nlinfo.nl_net,
|
||||||
|
fib_prefsrc, tb_id) != RTN_LOCAL) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
struct fib_info *fib_create_info(struct fib_config *cfg)
|
struct fib_info *fib_create_info(struct fib_config *cfg)
|
||||||
{
|
{
|
||||||
int err;
|
int err;
|
||||||
|
@ -1033,12 +1050,8 @@ struct fib_info *fib_create_info(struct fib_config *cfg)
|
||||||
fi->fib_flags |= RTNH_F_LINKDOWN;
|
fi->fib_flags |= RTNH_F_LINKDOWN;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (fi->fib_prefsrc) {
|
if (fi->fib_prefsrc && !fib_valid_prefsrc(cfg, fi->fib_prefsrc))
|
||||||
if (cfg->fc_type != RTN_LOCAL || !cfg->fc_dst ||
|
goto err_inval;
|
||||||
fi->fib_prefsrc != cfg->fc_dst)
|
|
||||||
if (inet_addr_type(net, fi->fib_prefsrc) != RTN_LOCAL)
|
|
||||||
goto err_inval;
|
|
||||||
}
|
|
||||||
|
|
||||||
change_nexthops(fi) {
|
change_nexthops(fi) {
|
||||||
fib_info_update_nh_saddr(net, nexthop_nh);
|
fib_info_update_nh_saddr(net, nexthop_nh);
|
||||||
|
|
Загрузка…
Ссылка в новой задаче