INIT_HLIST_NODE(&ifa->addr_lst);
        ifa->scope = cfg->scope;
        ifa->prefix_len = cfg->plen;
+       ifa->rt_priority = cfg->rt_priority;
        ifa->flags = cfg->ifa_flags;
        /* No need to add the TENTATIVE flag for addresses with NODAD */
        if (!(cfg->ifa_flags & IFA_F_NODAD))
 
        cfg.pfx = &addr;
        cfg.scope = ipv6_addr_scope(cfg.pfx);
+       cfg.rt_priority = 0;
 
        ift = ipv6_add_addr(idev, &cfg, block, NULL);
        if (IS_ERR(ift)) {
  */
 
 static void
-addrconf_prefix_route(struct in6_addr *pfx, int plen, struct net_device *dev,
-                     unsigned long expires, u32 flags, gfp_t gfp_flags)
+addrconf_prefix_route(struct in6_addr *pfx, int plen, u32 metric,
+                     struct net_device *dev, unsigned long expires,
+                     u32 flags, gfp_t gfp_flags)
 {
        struct fib6_config cfg = {
                .fc_table = l3mdev_fib_table(dev) ? : RT6_TABLE_PREFIX,
-               .fc_metric = IP6_RT_PRIO_ADDRCONF,
+               .fc_metric = metric ? : IP6_RT_PRIO_ADDRCONF,
                .fc_ifindex = dev->ifindex,
                .fc_expires = expires,
                .fc_dst_len = plen,
                                expires = jiffies_to_clock_t(rt_expires);
                        }
                        addrconf_prefix_route(&pinfo->prefix, pinfo->prefix_len,
-                                             dev, expires, flags, GFP_ATOMIC);
+                                             0, dev, expires, flags,
+                                             GFP_ATOMIC);
                }
                fib6_info_release(rt);
        }
        ifp = ipv6_add_addr(idev, cfg, true, extack);
        if (!IS_ERR(ifp)) {
                if (!(cfg->ifa_flags & IFA_F_NOPREFIXROUTE)) {
-                       addrconf_prefix_route(&ifp->addr, ifp->prefix_len, dev,
-                                             expires, flags, GFP_KERNEL);
+                       addrconf_prefix_route(&ifp->addr, ifp->prefix_len,
+                                             ifp->rt_priority, dev, expires,
+                                             flags, GFP_KERNEL);
                }
 
                /* Send a netlink notification if DAD is enabled and
 
        if (addr.s6_addr32[3]) {
                add_addr(idev, &addr, plen, scope);
-               addrconf_prefix_route(&addr, plen, idev->dev, 0, pflags,
+               addrconf_prefix_route(&addr, plen, 0, idev->dev, 0, pflags,
                                      GFP_ATOMIC);
                return;
        }
                                }
 
                                add_addr(idev, &addr, plen, flag);
-                               addrconf_prefix_route(&addr, plen, idev->dev, 0,
-                                                     pflags, GFP_ATOMIC);
+                               addrconf_prefix_route(&addr, plen, 0, idev->dev,
+                                                     0, pflags, GFP_ATOMIC);
                        }
                }
        }
 
        ifp = ipv6_add_addr(idev, &cfg, true, NULL);
        if (!IS_ERR(ifp)) {
-               addrconf_prefix_route(&ifp->addr, ifp->prefix_len, idev->dev,
+               addrconf_prefix_route(&ifp->addr, ifp->prefix_len, 0, idev->dev,
                                      0, 0, GFP_ATOMIC);
                addrconf_dad_start(ifp);
                in6_ifa_put(ifp);
                        addrconf_add_linklocal(idev, &addr,
                                               IFA_F_STABLE_PRIVACY);
                else if (prefix_route)
-                       addrconf_prefix_route(&addr, 64, idev->dev,
+                       addrconf_prefix_route(&addr, 64, 0, idev->dev,
                                              0, 0, GFP_KERNEL);
                break;
        case IN6_ADDR_GEN_MODE_EUI64:
                if (ipv6_generate_eui64(addr.s6_addr + 8, idev->dev) == 0)
                        addrconf_add_linklocal(idev, &addr, 0);
                else if (prefix_route)
-                       addrconf_prefix_route(&addr, 64, idev->dev,
+                       addrconf_prefix_route(&addr, 64, 0, idev->dev,
                                              0, 0, GFP_KERNEL);
                break;
        case IN6_ADDR_GEN_MODE_NONE:
 
        if (!(ifp->flags & IFA_F_NOPREFIXROUTE)) {
                addrconf_prefix_route(&ifp->addr, ifp->prefix_len,
-                                     idev->dev, 0, 0, GFP_ATOMIC);
+                                     ifp->rt_priority, idev->dev, 0, 0,
+                                     GFP_ATOMIC);
        }
 
        if (ifp->state == INET6_IFADDR_STATE_PREDAD)
        [IFA_LOCAL]             = { .len = sizeof(struct in6_addr) },
        [IFA_CACHEINFO]         = { .len = sizeof(struct ifa_cacheinfo) },
        [IFA_FLAGS]             = { .len = sizeof(u32) },
+       [IFA_RT_PRIORITY]       = { .len = sizeof(u32) },
 };
 
 static int
                              ifm->ifa_prefixlen);
 }
 
+static int modify_prefix_route(struct inet6_ifaddr *ifp,
+                              unsigned long expires, u32 flags)
+{
+       struct fib6_info *f6i;
+
+       f6i = addrconf_get_prefix_route(&ifp->addr,
+                                       ifp->prefix_len,
+                                       ifp->idev->dev,
+                                       0, RTF_GATEWAY | RTF_DEFAULT);
+       if (!f6i)
+               return -ENOENT;
+
+       if (f6i->fib6_metric != ifp->rt_priority) {
+               /* add new one */
+               addrconf_prefix_route(&ifp->addr, ifp->prefix_len,
+                                     ifp->rt_priority, ifp->idev->dev,
+                                     expires, flags, GFP_KERNEL);
+               /* delete old one */
+               ip6_del_rt(dev_net(ifp->idev->dev), f6i);
+       } else {
+               if (!expires)
+                       fib6_clean_expires(f6i);
+               else
+                       fib6_set_expires(f6i, expires);
+
+               fib6_info_release(f6i);
+       }
+
+       return 0;
+}
+
 static int inet6_addr_modify(struct inet6_ifaddr *ifp, struct ifa6_config *cfg)
 {
        u32 flags;
        ifp->valid_lft = cfg->valid_lft;
        ifp->prefered_lft = cfg->preferred_lft;
 
+       if (cfg->rt_priority && cfg->rt_priority != ifp->rt_priority)
+               ifp->rt_priority = cfg->rt_priority;
+
        spin_unlock_bh(&ifp->lock);
        if (!(ifp->flags&IFA_F_TENTATIVE))
                ipv6_ifa_notify(0, ifp);
 
        if (!(cfg->ifa_flags & IFA_F_NOPREFIXROUTE)) {
-               addrconf_prefix_route(&ifp->addr, ifp->prefix_len,
-                                     ifp->idev->dev, expires, flags,
-                                     GFP_KERNEL);
+               int rc = -ENOENT;
+
+               if (had_prefixroute)
+                       rc = modify_prefix_route(ifp, expires, flags);
+
+               /* prefix route could have been deleted; if so restore it */
+               if (rc == -ENOENT) {
+                       addrconf_prefix_route(&ifp->addr, ifp->prefix_len,
+                                             ifp->rt_priority, ifp->idev->dev,
+                                             expires, flags, GFP_KERNEL);
+               }
        } else if (had_prefixroute) {
                enum cleanup_prefix_rt_t action;
                unsigned long rt_expires;
 
        cfg.peer_pfx = peer_pfx;
        cfg.plen = ifm->ifa_prefixlen;
+       if (tb[IFA_RT_PRIORITY])
+               cfg.rt_priority = nla_get_u32(tb[IFA_RT_PRIORITY]);
+
        cfg.valid_lft = INFINITY_LIFE_TIME;
        cfg.preferred_lft = INFINITY_LIFE_TIME;
 
               + nla_total_size(16) /* IFA_LOCAL */
               + nla_total_size(16) /* IFA_ADDRESS */
               + nla_total_size(sizeof(struct ifa_cacheinfo))
-              + nla_total_size(4)  /* IFA_FLAGS */;
+              + nla_total_size(4)  /* IFA_FLAGS */
+              + nla_total_size(4)  /* IFA_RT_PRIORITY */;
 }
 
 static int inet6_fill_ifaddr(struct sk_buff *skb, struct inet6_ifaddr *ifa,
                if (nla_put_in6_addr(skb, IFA_ADDRESS, &ifa->addr) < 0)
                        goto error;
 
+       if (ifa->rt_priority &&
+           nla_put_u32(skb, IFA_RT_PRIORITY, ifa->rt_priority))
+               goto error;
+
        if (put_cacheinfo(skb, ifa->cstamp, ifa->tstamp, preferred, valid) < 0)
                goto error;
 
                if (ifp->idev->cnf.forwarding)
                        addrconf_join_anycast(ifp);
                if (!ipv6_addr_any(&ifp->peer_addr))
-                       addrconf_prefix_route(&ifp->peer_addr, 128,
+                       addrconf_prefix_route(&ifp->peer_addr, 128, 0,
                                              ifp->idev->dev, 0, 0,
                                              GFP_ATOMIC);
                break;