return err;
 }
 
-void rt6_flush_exceptions(struct fib6_info *rt)
+static void fib6_nh_flush_exceptions(struct fib6_nh *nh, struct fib6_info *from)
 {
        struct rt6_exception_bucket *bucket;
        struct rt6_exception *rt6_ex;
 
        spin_lock_bh(&rt6_exception_lock);
        /* Prevent rt6_insert_exception() to recreate the bucket list */
-       rt->exception_bucket_flushed = 1;
+       from->exception_bucket_flushed = 1;
 
-       bucket = rcu_dereference_protected(rt->rt6i_exception_bucket,
+       bucket = rcu_dereference_protected(from->rt6i_exception_bucket,
                                    lockdep_is_held(&rt6_exception_lock));
        if (!bucket)
                goto out;
        spin_unlock_bh(&rt6_exception_lock);
 }
 
+void rt6_flush_exceptions(struct fib6_info *f6i)
+{
+       fib6_nh_flush_exceptions(&f6i->fib6_nh, f6i);
+}
+
 /* Find cached rt in the hash table inside passed in rt
  * Caller has to hold rcu_read_lock()
  */
 }
 
 /* Remove the passed in cached rt from the hash table that contains it */
-static int rt6_remove_exception_rt(struct rt6_info *rt)
+static int fib6_nh_remove_exception(const struct fib6_info *from, int plen,
+                                   const struct rt6_info *rt)
 {
+       const struct in6_addr *src_key = NULL;
        struct rt6_exception_bucket *bucket;
-       struct in6_addr *src_key = NULL;
        struct rt6_exception *rt6_ex;
-       struct fib6_info *from;
        int err;
 
-       from = rcu_dereference(rt->from);
-       if (!from ||
-           !(rt->rt6i_flags & RTF_CACHE))
-               return -EINVAL;
-
        if (!rcu_access_pointer(from->rt6i_exception_bucket))
                return -ENOENT;
 
        bucket = rcu_dereference_protected(from->rt6i_exception_bucket,
                                    lockdep_is_held(&rt6_exception_lock));
 #ifdef CONFIG_IPV6_SUBTREES
-       /* rt6i_src.plen != 0 indicates 'from' is in subtree
-        * and exception table is indexed by a hash of
-        * both rt6i_dst and rt6i_src.
+       /* plen != 0 indicates 'from' is in subtree and exception
+        * table is indexed by a hash of both rt6i_dst and rt6i_src.
         * Otherwise, the exception table is indexed by
         * a hash of only rt6i_dst.
         */
-       if (from->fib6_src.plen)
+       if (plen)
                src_key = &rt->rt6i_src.addr;
 #endif
        rt6_ex = __rt6_find_exception_spinlock(&bucket,
        return err;
 }
 
+static int rt6_remove_exception_rt(struct rt6_info *rt)
+{
+       struct fib6_info *from;
+
+       from = rcu_dereference(rt->from);
+       if (!from ||
+           !(rt->rt6i_flags & RTF_CACHE))
+               return -EINVAL;
+
+       return fib6_nh_remove_exception(from, from->fib6_src.plen, rt);
+}
+
 /* Find rt6_ex which contains the passed in rt cache and
  * refresh its stamp
  */
-static void rt6_update_exception_stamp_rt(struct rt6_info *rt)
+static void fib6_nh_update_exception(const struct fib6_info *from, int plen,
+                                    const struct rt6_info *rt)
 {
+       const struct in6_addr *src_key = NULL;
        struct rt6_exception_bucket *bucket;
-       struct in6_addr *src_key = NULL;
        struct rt6_exception *rt6_ex;
-       struct fib6_info *from;
-
-       rcu_read_lock();
-       from = rcu_dereference(rt->from);
-       if (!from || !(rt->rt6i_flags & RTF_CACHE))
-               goto unlock;
 
        bucket = rcu_dereference(from->rt6i_exception_bucket);
 
 #ifdef CONFIG_IPV6_SUBTREES
-       /* rt6i_src.plen != 0 indicates 'from' is in subtree
-        * and exception table is indexed by a hash of
-        * both rt6i_dst and rt6i_src.
+       /* plen != 0 indicates 'from' is in subtree and exception
+        * table is indexed by a hash of both rt6i_dst and rt6i_src.
         * Otherwise, the exception table is indexed by
         * a hash of only rt6i_dst.
         */
-       if (from->fib6_src.plen)
+       if (plen)
                src_key = &rt->rt6i_src.addr;
 #endif
        rt6_ex = __rt6_find_exception_rcu(&bucket,
                                          src_key);
        if (rt6_ex)
                rt6_ex->stamp = jiffies;
+}
+
+static void rt6_update_exception_stamp_rt(struct rt6_info *rt)
+{
+       struct fib6_info *from;
 
+       rcu_read_lock();
+
+       from = rcu_dereference(rt->from);
+       if (!from || !(rt->rt6i_flags & RTF_CACHE))
+               goto unlock;
+
+       fib6_nh_update_exception(from, from->fib6_src.plen, rt);
 unlock:
        rcu_read_unlock();
 }
        gc_args->more++;
 }
 
-void rt6_age_exceptions(struct fib6_info *rt,
-                       struct fib6_gc_args *gc_args,
-                       unsigned long now)
+static void fib6_nh_age_exceptions(struct fib6_info *rt,
+                                  struct fib6_gc_args *gc_args,
+                                  unsigned long now)
 {
        struct rt6_exception_bucket *bucket;
        struct rt6_exception *rt6_ex;
        rcu_read_unlock_bh();
 }
 
+void rt6_age_exceptions(struct fib6_info *rt,
+                       struct fib6_gc_args *gc_args,
+                       unsigned long now)
+{
+       fib6_nh_age_exceptions(rt, gc_args, now);
+}
+
 /* must be called with rcu lock held */
 int fib6_table_lookup(struct net *net, struct fib6_table *table, int oif,
                      struct flowi6 *fl6, struct fib6_result *res, int strict)
 struct rt6_mtu_change_arg {
        struct net_device *dev;
        unsigned int mtu;
+       struct fib6_info *f6i;
 };
 
-static int rt6_mtu_change_route(struct fib6_info *rt, void *p_arg)
+static int fib6_nh_mtu_change(struct fib6_info *f6i, void *_arg)
+{
+       struct rt6_mtu_change_arg *arg = (struct rt6_mtu_change_arg *)_arg;
+       struct fib6_nh *nh = &f6i->fib6_nh;
+
+       /* For administrative MTU increase, there is no way to discover
+        * IPv6 PMTU increase, so PMTU increase should be updated here.
+        * Since RFC 1981 doesn't include administrative MTU increase
+        * update PMTU increase is a MUST. (i.e. jumbo frame)
+        */
+       if (nh->fib_nh_dev == arg->dev) {
+               struct inet6_dev *idev = __in6_dev_get(arg->dev);
+               u32 mtu = f6i->fib6_pmtu;
+
+               if (mtu >= arg->mtu ||
+                   (mtu < arg->mtu && mtu == idev->cnf.mtu6))
+                       fib6_metric_set(f6i, RTAX_MTU, arg->mtu);
+
+               spin_lock_bh(&rt6_exception_lock);
+               rt6_exceptions_update_pmtu(idev, f6i, arg->mtu);
+               spin_unlock_bh(&rt6_exception_lock);
+       }
+
+       return 0;
+}
+
+static int rt6_mtu_change_route(struct fib6_info *f6i, void *p_arg)
 {
        struct rt6_mtu_change_arg *arg = (struct rt6_mtu_change_arg *) p_arg;
        struct inet6_dev *idev;
        if (!idev)
                return 0;
 
-       /* For administrative MTU increase, there is no way to discover
-          IPv6 PMTU increase, so PMTU increase should be updated here.
-          Since RFC 1981 doesn't include administrative MTU increase
-          update PMTU increase is a MUST. (i.e. jumbo frame)
-        */
-       if (rt->fib6_nh.fib_nh_dev == arg->dev &&
-           !fib6_metric_locked(rt, RTAX_MTU)) {
-               u32 mtu = rt->fib6_pmtu;
-
-               if (mtu >= arg->mtu ||
-                   (mtu < arg->mtu && mtu == idev->cnf.mtu6))
-                       fib6_metric_set(rt, RTAX_MTU, arg->mtu);
+       if (fib6_metric_locked(f6i, RTAX_MTU))
+               return 0;
 
-               spin_lock_bh(&rt6_exception_lock);
-               rt6_exceptions_update_pmtu(idev, rt, arg->mtu);
-               spin_unlock_bh(&rt6_exception_lock);
-       }
-       return 0;
+       arg->f6i = f6i;
+       return fib6_nh_mtu_change(f6i, arg);
 }
 
 void rt6_mtu_change(struct net_device *dev, unsigned int mtu)